diff --git a/404.html b/404.html
index aada469b..d2b6f276 100644
--- a/404.html
+++ b/404.html
@@ -26,7 +26,7 @@
-
+
Your browser is out of date!
-
+
diff --git a/assets/images/contact.svg b/assets/images/contact.svg
new file mode 100644
index 00000000..1f4409b5
--- /dev/null
+++ b/assets/images/contact.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/assets/images/landing/Legacy-SCK.gif b/assets/images/landing/Legacy-SCK.gif
new file mode 100755
index 00000000..924c9d03
Binary files /dev/null and b/assets/images/landing/Legacy-SCK.gif differ
diff --git a/assets/images/landing/SCK-Platform-mobile.jpg b/assets/images/landing/SCK-Platform-mobile.jpg
new file mode 100644
index 00000000..816bb13c
Binary files /dev/null and b/assets/images/landing/SCK-Platform-mobile.jpg differ
diff --git a/assets/images/landing/SCK-Platform.jpg b/assets/images/landing/SCK-Platform.jpg
new file mode 100644
index 00000000..6aa3060b
Binary files /dev/null and b/assets/images/landing/SCK-Platform.jpg differ
diff --git a/assets/images/landing/SCK-customized-projects.jpg b/assets/images/landing/SCK-customized-projects.jpg
new file mode 100755
index 00000000..515181d9
Binary files /dev/null and b/assets/images/landing/SCK-customized-projects.jpg differ
diff --git a/assets/images/landing/educators.jpg b/assets/images/landing/educators.jpg
new file mode 100644
index 00000000..0a02c1e9
Binary files /dev/null and b/assets/images/landing/educators.jpg differ
diff --git a/assets/images/landing/local-communites.jpg b/assets/images/landing/local-communites.jpg
new file mode 100644
index 00000000..8511df82
Binary files /dev/null and b/assets/images/landing/local-communites.jpg differ
diff --git a/assets/images/landing/research-institutions.jpg b/assets/images/landing/research-institutions.jpg
new file mode 100755
index 00000000..fedef4fd
Binary files /dev/null and b/assets/images/landing/research-institutions.jpg differ
diff --git a/assets/images/landing/smart-citizen-2-2-kit.gif b/assets/images/landing/smart-citizen-2-2-kit.gif
new file mode 100755
index 00000000..05567dab
Binary files /dev/null and b/assets/images/landing/smart-citizen-2-2-kit.gif differ
diff --git a/assets/images/landing/smart-citizen-2-2-kits-various.jpg b/assets/images/landing/smart-citizen-2-2-kits-various.jpg
new file mode 100755
index 00000000..f99e5ac8
Binary files /dev/null and b/assets/images/landing/smart-citizen-2-2-kits-various.jpg differ
diff --git a/assets/images/landing/smart-citizen-data.jpg b/assets/images/landing/smart-citizen-data.jpg
new file mode 100644
index 00000000..a921bf50
Binary files /dev/null and b/assets/images/landing/smart-citizen-data.jpg differ
diff --git a/assets/images/landing/smartcitizen-docs.jpg b/assets/images/landing/smartcitizen-docs.jpg
new file mode 100644
index 00000000..55b12fe2
Binary files /dev/null and b/assets/images/landing/smartcitizen-docs.jpg differ
diff --git a/assets/images/landing/smartcitizen-platform-mobile.jpg b/assets/images/landing/smartcitizen-platform-mobile.jpg
new file mode 100644
index 00000000..4463476e
Binary files /dev/null and b/assets/images/landing/smartcitizen-platform-mobile.jpg differ
diff --git a/assets/images/landing/smartcitizen-platform.jpg b/assets/images/landing/smartcitizen-platform.jpg
new file mode 100644
index 00000000..e6483a53
Binary files /dev/null and b/assets/images/landing/smartcitizen-platform.jpg differ
diff --git a/assets/images/landing/smartcitizen-water-station.jpg b/assets/images/landing/smartcitizen-water-station.jpg
new file mode 100644
index 00000000..967326db
Binary files /dev/null and b/assets/images/landing/smartcitizen-water-station.jpg differ
diff --git a/assets/images/landing/smartcitizen-workshop.jpg b/assets/images/landing/smartcitizen-workshop.jpg
new file mode 100644
index 00000000..4d6d227e
Binary files /dev/null and b/assets/images/landing/smartcitizen-workshop.jpg differ
diff --git a/assets/images/new.svg b/assets/images/new.svg
new file mode 100644
index 00000000..6e333234
--- /dev/null
+++ b/assets/images/new.svg
@@ -0,0 +1,82 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/index.html b/index.html
index aada469b..d2b6f276 100644
--- a/index.html
+++ b/index.html
@@ -26,7 +26,7 @@
-
+
Your browser is out of date!
-
+
diff --git a/scripts/app-8b845ff230.js b/scripts/app-5bbe3bafb5.js
similarity index 96%
rename from scripts/app-8b845ff230.js
rename to scripts/app-5bbe3bafb5.js
index 348ba523..01864a52 100644
--- a/scripts/app-8b845ff230.js
+++ b/scripts/app-5bbe3bafb5.js
@@ -9,2430 +9,2305 @@
'use strict';
angular.module('app.components')
- .controller('KitController', KitController);
+ .factory('User', ['COUNTRY_CODES', function(COUNTRY_CODES) {
- KitController.$inject = ['$state','$scope', '$stateParams',
- 'sensor', 'FullDevice', '$mdDialog', 'belongsToUser',
- 'timeUtils', 'animation', 'auth',
- '$timeout', 'alert', '$q', 'device',
- 'HasSensorDevice', 'geolocation', 'PreviewDevice'];
- function KitController($state, $scope, $stateParams,
- sensor, FullDevice, $mdDialog, belongsToUser,
- timeUtils, animation, auth,
- $timeout, alert, $q, device,
- HasSensorDevice, geolocation, PreviewDevice) {
+ /**
+ * User constructor
+ * @param {Object} userData - User data sent from API
+ * @property {number} id - User ID
+ * @property {string} username - Username
+ * @property {string} profile_picture - Avatar URL of user
+ * @property {Array} devices - Kits that belongs to this user
+ * @property {string} url - URL
+ * @property {string} city - User city
+ * @property {string} country - User country
+ */
- var vm = this;
- var sensorsData = [];
+ function User(userData) {
+ this.id = userData.id;
+ this.username = userData.username;
+ this.profile_picture = userData.profile_picture;
+ this.devices = userData.devices;
+ this.url = userData.url;
+ this.city = userData.location.city;
+ /*jshint camelcase: false */
+ this.country = COUNTRY_CODES[userData.location.country_code];
+ }
+ return User;
+ }]);
- var mainSensorID, compareSensorID;
- var picker;
- vm.deviceID = $stateParams.id;
- vm.battery = {};
- vm.downloadData = downloadData;
- vm.geolocate = geolocate;
- vm.device = undefined;
- vm.deviceBelongsToUser = belongsToUser;
- vm.deviceWithoutData = false;
- vm.legacyApiKey = belongsToUser ?
- auth.getCurrentUser().data.key :
- undefined;
- vm.loadingChart = true;
- vm.moveChart = moveChart;
- vm.allowUpdateChart = true;
- vm.ownerDevices = [];
- vm.removeDevice = removeDevice;
- vm.resetTimeOpts = resetTimeOpts;
- vm.sampleDevices = [];
- vm.selectedSensor = undefined;
- vm.selectedSensorData = {};
- vm.selectedSensorToCompare = undefined;
- vm.selectedSensorToCompareData = {};
- vm.sensors = [];
- vm.chartSensors = [];
- vm.sensorsToCompare = [];
- vm.setFromLast = setFromLast;
- vm.showSensorOnChart = showSensorOnChart;
- vm.showStore = showStore;
- vm.slide = slide;
- vm.showRaw = false;
- vm.timeOpt = ['60 minutes', 'day' , 'month'];
- vm.timeOptSelected = timeOptSelected;
- vm.updateInterval = 15000;
- vm.hasRaw;
- vm.sensorNames = {};
+})();
- var focused = true;
+(function() {
+ 'use strict';
- // event listener on change of value of main sensor selector
- $scope.$watch('vm.selectedSensor', function(newVal) {
+ angular.module('app.components')
+ .factory('NonAuthUser', ['User', function(User) {
- // Prevents undisered calls if selected sensor is not yet defined
- if (!newVal) {
- return;
+ function NonAuthUser(userData) {
+ User.call(this, userData);
}
+ NonAuthUser.prototype = Object.create(User.prototype);
+ NonAuthUser.prototype.constructor = User;
- vm.selectedSensorToCompare = undefined;
- vm.selectedSensorToCompareData = {};
- vm.chartDataCompare = [];
- compareSensorID = undefined;
+ return NonAuthUser;
+ }]);
+})();
- setSensorSideChart();
+(function() {
+ 'use strict';
- vm.sensorsToCompare = getSensorsToCompare();
+ angular.module('app.components')
+ .factory('AuthUser', ['User', function(User) {
- $timeout(function() {
- // TODO: Improvement, change how we set the colors
- colorSensorCompareName();
- setSensor({type: 'main', value: newVal});
+ /**
+ * AuthUser constructor. Used for authenticated users
+ * @extends User
+ * @param {Object} userData - Contains user data sent from API
+ * @property {string} email - User email
+ * @property {string} role - User role. Ex: admin
+ * @property {string} key - Personal API Key
+ */
- if (picker){
- changeChart([mainSensorID]);
- }
- }, 100);
+ function AuthUser(userData) {
+ User.call(this, userData);
- });
+ this.email = userData.email;
+ this.role = userData.role;
+ /*jshint camelcase: false */
+ this.key = userData.legacy_api_key;
+ }
+ AuthUser.prototype = Object.create(User.prototype);
+ AuthUser.prototype.constructor = User;
- // event listener on change of value of compare sensor selector
- $scope.$watch('vm.selectedSensorToCompare', function(newVal, oldVal) {
- vm.sensorsToCompare.forEach(function(sensor) {
- if(sensor.id === newVal) {
- _.extend(vm.selectedSensorToCompareData, sensor);
- }
- });
+ return AuthUser;
+ }]);
+})();
- $timeout(function() {
- colorSensorCompareName();
- setSensor({type: 'compare', value: newVal});
+(function() {
+ 'use strict';
- if(oldVal === undefined && newVal === undefined) {
- return;
+ angular.module('app.components')
+ .factory('Sensor', ['sensorUtils', 'timeUtils', function(sensorUtils, timeUtils) {
+
+ /**
+ * Sensor constructor
+ * @param {Object} sensorData - Contains the data of a sensor sent from the API
+ * @property {string} name - Name of sensor
+ * @property {number} id - ID of sensor
+ * @property {string} unit - Unit of sensor. Ex: %
+ * @property {string} value - Last value sent. Ex: 95
+ * @property {string} prevValue - Previous value before last value
+ * @property {string} lastReadingAt - last_reading_at for the sensor reading
+ * @property {string} icon - Icon URL for sensor
+ * @property {string} arrow - Icon URL for sensor trend(up, down or equal)
+ * @property {string} color - Color that belongs to sensor
+ * @property {object} measurement - Measurement
+ * @property {string} fullDescription - Full Description for popup
+ * @property {string} previewDescription - Short Description for dashboard. Max 140 chars
+ * @property {string} tags - Contains sensor tags for filtering the view
+ */
+ function Sensor(sensorData) {
+
+ this.id = sensorData.id;
+ this.name = sensorData.name;
+ this.unit = sensorData.unit;
+ this.value = sensorUtils.getSensorValue(sensorData);
+ this.prevValue = sensorUtils.getSensorPrevValue(sensorData);
+ this.lastReadingAt = timeUtils.parseDate(sensorData.last_reading_at);
+ this.icon = sensorUtils.getSensorIcon(this.name);
+ this.arrow = sensorUtils.getSensorArrow(this.value, this.prevValue);
+ this.color = sensorUtils.getSensorColor(this.name);
+ this.measurement = sensorData.measurement;
+
+ // Some sensors don't have measurements because they are ancestors
+ if (sensorData.measurement) {
+ var description = sensorData.measurement.description;
+ this.fullDescription = description;
+ this.previewDescription = description.length > 140 ? description.slice(
+ 0, 140).concat(' ... ') : description;
+ this.is_ancestor = false;
+ } else {
+ this.is_ancestor = true;
}
- changeChart([compareSensorID]);
- }, 100);
- });
+ // Get sensor tags
+ this.tags = sensorData.tags;
+ }
- $scope.$on('hideChartSpinner', function() {
- vm.loadingChart = false;
- });
+ return Sensor;
+ }]);
+})();
+(function() {
+ 'use strict';
- $scope.$on('$destroy', function() {
- focused = false;
- $timeout.cancel(vm.updateTimeout);
- });
+ angular.module('app.components')
+ .factory('SearchResultLocation', ['SearchResult', function(SearchResult) {
- $scope.$on('$viewContentLoaded', function(event){
- initialize();
- });
+ /**
+ * Search Result Location constructor
+ * @extends SearchResult
+ * @param {Object} object - Object that contains the search result data from API
+ * @property {number} lat - Latitude
+ * @property {number} lng - Longitude
+ */
+ function SearchResultLocation(object) {
+ SearchResult.call(this, object);
- function initialize() {
- animation.viewLoaded();
- updatePeriodically();
- }
+ this.lat = object.latitude;
+ this.lng = object.longitude;
+ this.layer = object.layer;
+ }
+ return SearchResultLocation;
+ }]);
- function pollAndUpdate(){
- vm.updateTimeout = $timeout(function() {
- updatePeriodically();
- }, vm.updateInterval);
- }
+})();
- function updatePeriodically(){
- getAndUpdateDevice().then(function(){
- pollAndUpdate();
- });
- }
+(function() {
+ 'use strict';
- function getAndUpdateDevice(){
- // TODO: Improvement UX Change below to && to avoid constant unhandled error
- // Through reject is not possible
- if (vm.deviceID || !isNaN(vm.deviceID)){
- return device.getDevice(vm.deviceID)
- .then(function(deviceData) {
- if (deviceData.is_private) {
- deviceIsPrivate();
- }
- var newDevice = new FullDevice(deviceData);
- vm.prevDevice = vm.device;
+ angular.module('app.components')
+ .factory('SearchResult', ['searchUtils', function(searchUtils) {
- if (vm.prevDevice) {
- /* Kit already loaded. We are waiting for updates */
- if (vm.prevDevice.state.name !== 'has published' && newDevice.state.name === 'has published'){
- /* The kit has just published data for the first time. Fully reload the view */
- return $q.reject({justPublished: true});
- } else if(new Date(vm.prevDevice.lastReadingAt.raw) >= new Date(newDevice.lastReadingAt.raw)) {
- /* Break if there's no new data*/
- return $q.reject();
- }
- }
+ /**
+ * Search Result constructor
+ * @param {Object} object - Object that belongs to a search result from API
+ * @property {string} type - Type of search result. Ex: Country, City, User, Device
+ * @property {number} id - ID of search result, only for user & device
+ * @property {string} name - Name of search result, only for user & device
+ * @property {string} location - Location of search result. Ex: 'Paris, France'
+ * @property {string} icon - URL for the icon that belongs to this search result
+ * @property {string} iconType - Type of icon. Can be either img or div
+ */
+
+ function SearchResult(object) {
+ this.type = object.type;
+ this.id = object.id;
+ this.name = searchUtils.parseName(object);
+ this.location = searchUtils.parseLocation(object);
+ this.icon = searchUtils.parseIcon(object, this.type);
+ this.iconType = searchUtils.parseIconType(this.type);
+ }
+ return SearchResult;
+ }]);
+})();
- vm.device = newDevice;
- setOwnerSampleDevices();
+(function() {
+ 'use strict';
- if (vm.device.state.name === 'has published') {
- /* Device has data */
- setDeviceOnMap();
- setChartTimeRange();
- deviceAnnouncements();
+ angular.module('app.components')
+ .factory('Marker', ['deviceUtils', 'markerUtils', 'timeUtils', '$state', function(deviceUtils, markerUtils, timeUtils, $state) {
+ /**
+ * Marker constructor
+ * @constructor
+ * @param {Object} deviceData - Object with data about marker from API
+ * @property {number} lat - Latitude
+ * @property {number} lng - Longitude
+ * @property {string} message - Message inside marker popup
+ * @property {Object} icon - Object with classname, size and type of marker icon
+ * @property {string} layer - Map layer that icons belongs to
+ * @property {boolean} focus - Whether marker popup is opened
+ * @property {Object} myData - Marker id and labels
+ */
+ function Marker(deviceData) {
+ let linkStart = '', linkEnd = '';
+ const id = deviceData.id;
+ if ($state.$current.name === 'embbed') {
+ linkStart = '';
+ linkEnd = ' ';
+ }
+ this.lat = deviceUtils.parseCoordinates(deviceData).lat;
+ this.lng = deviceUtils.parseCoordinates(deviceData).lng;
+ // TODO: Bug, pop-up lastreading at doesn't get updated by publication
+ this.message = '
';
- /*Load sensor if it has already published*/
- return $q.all([getMainSensors(vm.device), getCompareSensors(vm.device)]);
- } else {
- /* Device just loaded and has no data yet */
- return $q.reject({noSensorData: true});
- }
- })
- .then(setSensors, killSensorsLoading);
+ this.icon = markerUtils.getIcon(deviceData);
+ this.layer = 'devices';
+ this.focus = false;
+ this.myData = {
+ id: id,
+ labels: deviceUtils.parseSystemTags(deviceData),
+ tags: deviceUtils.parseUserTags(deviceData)
+ };
}
- }
+ return Marker;
- function killSensorsLoading(error){
- if(error) {
- if(error.status === 404) {
- $state.go('layout.404');
- }
- else if (error.justPublished) {
- $state.transitionTo($state.current, {reloadMap: true, id: vm.deviceID}, {
- reload: true, inherit: false, notify: true
- });
- }
- else if (error.noSensorData) {
- deviceHasNoData();
+ function createTagsTemplate(tagsArr, tagType, clickable) {
+ if(typeof(clickable) === 'undefined'){
+ clickable = false;
}
- else if (error.status === 403){
- deviceIsPrivate();
+ var clickablTag = '';
+ if(clickable){
+ clickablTag = 'clickable';
}
- }
- }
- function deviceAnnouncements(){
- if(!timeUtils.isWithin(1, 'months', vm.device.lastReadingAt.raw)) {
- //TODO: Cosmetic Update the message
- alert.info.longTime();
- }
- /* The device has just published data after not publishing for 15min */
- else if(vm.prevDevice && timeUtils.isDiffMoreThan15min(vm.prevDevice.lastReadingAt.raw, vm.device.lastReadingAt.raw)) {
- alert.success('Your Kit just published again!');
- }
- }
+ if(!tagType){
+ tagType = 'tag';
+ }
- function deviceHasNoData() {
- vm.deviceWithoutData = true;
- animation.deviceWithoutData({device: vm.device, belongsToUser:vm.deviceBelongsToUser});
- if(vm.deviceBelongsToUser) {
- alert.info.noData.owner($stateParams.id);
- } else {
- alert.info.noData.visitor();
+ return _.reduce(tagsArr, function(acc, label) {
+ var element ='';
+ if(tagType === 'tag'){
+ element = ' ';
+ }else{
+ element = ''+label+' ';
+ }
+ return acc.concat(element);
+ }, '');
}
- }
- function deviceIsPrivate() {
- alert.info.noData.private();
- }
+ }]);
+})();
- function setOwnerSampleDevices() {
- // TODO: Refactor - this information is in the user, no need to go to devices
- getOwnerDevices(vm.device, -6)
- .then(function(ownerDevices){
- vm.sampleDevices = ownerDevices;
- });
- }
+(function () {
+ 'use strict';
- function setChartTimeRange() {
- if(vm.allowUpdateChart) {
- /* Init the chart range to default if doesn't exist of the user hasn't interacted */
- picker = initializePicker();
- }
- }
+ angular.module('app.components')
+ .factory('PreviewDevice', ['Device', function (Device) {
- function setDeviceOnMap() {
- animation.deviceLoaded({lat: vm.device.latitude, lng: vm.device.longitude,
- id: vm.device.id});
- }
+ /**
+ * Preview Device constructor.
+ * Used for devices stacked in a list, like in User Profile or Device states
+ * @extends Device
+ * @constructor
+ * @param {Object} object - Object with all the data about the device from the API
+ */
+ function PreviewDevice(object) {
+ Device.call(this, object);
- function setSensors(sensorsRes){
+ this.dropdownOptions = [];
+ this.dropdownOptions.push({ text: 'EDIT', value: '1', href: 'kits/' + this.id + '/edit', icon: 'fa fa-edit' });
+ this.dropdownOptions.push({ text: 'SD CARD UPLOAD', value: '2', href: 'kits/' + this.id + '/upload', icon: 'fa fa-sd-card' });
+ }
+ PreviewDevice.prototype = Object.create(Device.prototype);
+ PreviewDevice.prototype.constructor = Device;
+ return PreviewDevice;
+ }]);
+})();
- var mainSensors = sensorsRes[0];
- var compareSensors = sensorsRes[1];
+(function() {
+ 'use strict';
- vm.battery = _.find(mainSensors, {name: 'battery'});
- vm.sensors = mainSensors.reverse();
- vm.sensors.forEach(checkRaw);
- vm.sensors.forEach(getHardwareName);
+ angular.module('app.components')
+ .factory('HasSensorDevice', ['Device', function(Device) {
- setSensorSideChart();
+ function HasSensorDevice(object) {
+ Device.call(this, object);
- if (!vm.selectedSensor) {
- vm.chartSensors = vm.sensors;
- vm.sensorsToCompare = compareSensors;
- vm.selectedSensor = (vm.sensors && vm.sensors[0]) ? vm.sensors[0].id : undefined;
+ this.sensors = object.data.sensors;
+ this.longitude = object.data.location.longitude;
+ this.latitude = object.data.location.latitude;
}
- animation.mapStateLoaded();
- }
+ HasSensorDevice.prototype = Object.create(Device.prototype);
+ HasSensorDevice.prototype.constructor = Device;
- function checkRaw(value){
- vm.hasRaw |= (value.tags.indexOf('raw') !== -1);
- }
+ HasSensorDevice.prototype.sensorsHasData = function() {
+ var parsedSensors = this.sensors.map(function(sensor) {
+ return sensor.value;
+ });
- function getHardwareName(value) {
- vm.sensorNames[value.id] = vm.device.sensors.find(element => element.id === value.id).name;
- }
- function setSensorSideChart() {
- if(vm.sensors){
- vm.sensors.forEach(function(sensor) {
- if(sensor.id === vm.selectedSensor) {
- _.extend(vm.selectedSensorData, sensor);
- }
+ return _.some(parsedSensors, function(sensorValue) {
+ return !!sensorValue;
});
- }
- }
+ };
- function removeDevice() {
- var confirm = $mdDialog.confirm()
- .title('Delete this kit?')
- .textContent('Are you sure you want to delete this kit?')
- .ariaLabel('')
- .ok('DELETE')
- .cancel('Cancel')
- .theme('primary')
- .clickOutsideToClose(true);
+ return HasSensorDevice;
+ }]);
+})();
- $mdDialog
- .show(confirm)
- .then(function(){
- device
- .removeDevice(vm.device.id)
- .then(function(){
- alert.success('Your kit was deleted successfully');
- device.updateContext().then(function(){
- $state.transitionTo('layout.myProfile.kits', $stateParams,
- { reload: false,
- inherit: false,
- notify: true
- });
- });
- })
- .catch(function(){
- alert.error('Error trying to delete your kit.');
- });
- });
- }
+(function() {
+ 'use strict';
- function showSensorOnChart(sensorID) {
- vm.selectedSensor = sensorID;
- }
+ angular.module('app.components')
+ .factory('FullDevice', ['Device', 'Sensor', 'deviceUtils', function(Device, Sensor, deviceUtils) {
- function slide(direction) {
- var slideContainer = angular.element('.sensors_container');
- var scrollPosition = slideContainer.scrollLeft();
- var width = slideContainer.width();
- var slideStep = width/2;
+ /**
+ * Full Device constructor.
+ * @constructor
+ * @extends Device
+ * @param {Object} object - Object with all the data about the device from the API
+ * @property {Object} owner - Device owner data
+ * @property {Array} data - Device sensor's data
+ * @property {Array} sensors - Device sensors data
+ * @property {Array} postProcessing - Device postprocessing
+ */
+ function FullDevice(object) {
+ Device.call(this, object);
- if(direction === 'left') {
- slideContainer.animate({'scrollLeft': scrollPosition + slideStep},
- {duration: 250, queue:false});
- } else if(direction === 'right') {
- slideContainer.animate({'scrollLeft': scrollPosition - slideStep},
- {duration: 250, queue:false});
+ this.owner = deviceUtils.parseOwner(object);
+ this.postProcessing = object.postprocessing;
+ this.data = object.data;
+ this.sensors = object.data.sensors;
}
- }
- function getSensorsToCompare() {
- return vm.sensors ? vm.sensors.filter(function(sensor) {
- return sensor.id !== vm.selectedSensor;
- }) : [];
- }
+ FullDevice.prototype = Object.create(Device.prototype);
+ FullDevice.prototype.constructor = FullDevice;
- function changeChart(sensorsID, options) {
- if(!sensorsID[0]) {
- return;
- }
+ FullDevice.prototype.getSensors = function(options) {
+ var sensors = _(this.data.sensors)
+ .chain()
+ .map(function(sensor) {
+ return new Sensor(sensor);
+ }).sort(function(a, b) {
+ /* This is a temporary hack to set always PV panel at the end*/
+ if (a.id === 18){ return -1;}
+ if (b.id === 18){ return 1;}
+ /* This is a temporary hack to set always the Battery at the end*/
+ if (a.id === 17){ return -1;}
+ if (b.id === 17){ return 1;}
+ /* This is a temporary hack to set always the Battery at the end*/
+ if (a.id === 10){ return -1;}
+ if (b.id === 10){ return 1;}
+ /* After the hacks, sort the sensors by id */
+ return b.id - a.id;
+ })
+ .tap(function(sensors) {
+ if(options.type === 'compare') {
+ sensors.unshift({
+ name: 'NONE',
+ color: 'white',
+ id: -1
+ });
+ }
+ })
+ .value();
+ return sensors;
+ };
- if(!options) {
- options = {};
- }
- options.from = options && options.from || picker.getValuePickerFrom();
- options.to = options && options.to || picker.getValuePickerTo();
+ return FullDevice;
+ }]);
+})();
- //show spinner
- vm.loadingChart = true;
- //grab chart data and save it
+(function() {
+ 'use strict';
- // it can be either 2 sensors or 1 sensor, so we use $q.all to wait for all
- $q.all(
- _.map(sensorsID, function(sensorID) {
- return getChartData($stateParams.id, sensorID, options.from, options.to)
- .then(function(data) {
- return data;
- });
- })
- ).then(function() {
- // after all sensors resolve, prepare data and attach it to scope
- // the variable on the scope will pass the data to the chart directive
- vm.chartDataMain = prepareChartData([mainSensorID, compareSensorID]);
- });
- }
- // calls api to get sensor data and saves it to sensorsData array
- function getChartData(deviceID, sensorID, dateFrom, dateTo, options) {
- return sensor.getSensorsData(deviceID, sensorID, dateFrom, dateTo)
- .then(function(data) {
- //save sensor data of this kit so that it can be reused
- sensorsData[sensorID] = data.readings;
- return data;
- });
- }
+ angular.module('app.components')
+ .factory('Device', ['deviceUtils', 'timeUtils', function(deviceUtils, timeUtils) {
- function prepareChartData(sensorsID) {
- var compareSensor;
- var parsedDataMain = parseSensorData(sensorsData, sensorsID[0]);
- var mainSensor = {
- data: parsedDataMain,
- color: vm.selectedSensorData.color,
- unit: vm.selectedSensorData.unit
- };
- if(sensorsID[1] && sensorsID[1] !== -1) {
- var parsedDataCompare = parseSensorData(sensorsData, sensorsID[1]);
+ /**
+ * Device constructor.
+ * @constructor
+ * @param {Object} object - Object with all the data about the device from the API
+ * @property {number} id - ID of the device
+ * @property {string} name - Name of the device
+ * @property {string} state - State of the device. Ex: Never published
+ * @property {string} description - Device description
+ * @property {Array} systemTags - System tags
+ * @property {Array} userTags - User tags. Ex: ''
+ * @property {bool} isPrivate - True if private device
+ * @property {Array} notifications - Notifications for low battery and stopped publishing
+ * @property {Object} lastReadingAt - last_reading_at: raw, ago, and parsed
+ * @property {Object} createdAt - created_at: raw, ago, and parsed
+ * @property {Object} updatedAt - updated_at: raw, ago, and parsed
+ * @property {Object} location - Location of device. Object with lat, long, elevation, city, country, country_code
+ * @property {string} locationString - Location of device. Ex: Madrid, Spain; Germany; Paris, France
+ * @property {Object} hardware - Device hardware field. Contains type, version, info, slug and name
+ * @property {string} hardwareName - Device hardware name
+ * @property {bool} isLegacy - True if legacy device
+ * @property {bool} isSCK - True if SC device
+ * @property {string} avatar - URL that contains the user avatar
+ */
+ function Device(object) {
+ // Basic information
+ this.id = object.id;
+ this.name = object.name;
+ this.state = deviceUtils.parseState(object);
+ this.description = object.description;
+ this.token = object.device_token;
+ this.macAddress = object.mac_address;
- compareSensor = {
- data: parsedDataCompare,
- color: vm.selectedSensorToCompareData.color,
- unit: vm.selectedSensorToCompareData.unit
- };
- }
- var newChartData = [mainSensor, compareSensor];
- return newChartData;
- }
+ // Tags and dates
+ this.systemTags = deviceUtils.parseSystemTags(object);
+ this.userTags = deviceUtils.parseUserTags(object);
+ this.isPrivate = deviceUtils.isPrivate(object);
+ this.preciseLocation = deviceUtils.preciseLocation(object);
+ this.enableForwarding = deviceUtils.enableForwarding(object);
+ this.notifications = deviceUtils.parseNotifications(object);
+ this.lastReadingAt = timeUtils.parseDate(object.last_reading_at);
+ this.createdAt = timeUtils.parseDate(object.created_at);
+ this.updatedAt = timeUtils.parseDate(object.updated_at);
- function parseSensorData(data, sensorID) {
- if(data.length === 0) {
- return [];
- }
- return data[sensorID].map(function(dataPoint) {
- var time = timeUtils.formatDate(dataPoint[0]);
- var value = dataPoint[1];
- var count = value === null ? 0 : value;
- return {
- time: time,
- count: count,
- value: value
- };
- });
- }
+ // Location
+ this.location = object.location;
+ this.locationString = deviceUtils.parseLocation(object);
- function setSensor(options) {
- var sensorID = options.value;
- if(sensorID === undefined) {
- return;
- }
- if(options.type === 'main') {
- mainSensorID = sensorID;
- } else if(options.type === 'compare') {
- compareSensorID = sensorID;
+ // Hardware
+ this.hardware = deviceUtils.parseHardware(object);
+ this.hardwareName = deviceUtils.parseHardwareName(this);
+ this.isLegacy = deviceUtils.isLegacyVersion(this);
+ this.isSCK = deviceUtils.isSCKHardware(this);
+ // this.class = deviceUtils.classify(object); // TODO - Do we need this?
+
+ this.avatar = deviceUtils.parseAvatar();
+ /*jshint camelcase: false */
}
- }
- function colorSensorCompareName() {
- var name = angular.element('.sensor_compare').find('md-select-label').find('span');
- name.css('color', vm.selectedSensorToCompareData.color || 'white');
- var icon = angular.element('.sensor_compare').find('md-select-label').find('.md-select-icon');
- icon.css('color', 'white');
- }
+ return Device;
+ }]);
+})();
- function getCurrentRange() {
- var to = moment(picker.getValuePickerTo());
- var from = moment(picker.getValuePickerFrom());
- return to.diff(from)/1000;
- }
+(function() {
+ 'use strict';
- function moveChart(direction) {
+ angular.module('app.components')
+ .directive('noDataBackdrop', noDataBackdrop);
- var valueTo, valueFrom;
- //grab current date range
- var currentRange = getCurrentRange();
+ /**
+ * Backdrop for chart section when kit has no data
+ *
+ */
+ noDataBackdrop.$inject = [];
- /*jshint camelcase: false*/
- var from_picker = angular.element('#picker_from').pickadate('picker');
- var to_picker = angular.element('#picker_to').pickadate('picker');
+ function noDataBackdrop() {
+ return {
+ restrict: 'A',
+ scope: {},
+ templateUrl: 'app/core/animation/backdrop/noDataBackdrop.html',
+ controller: function($scope, $timeout) {
+ var vm = this;
- if(direction === 'left') {
- //set both from and to pickers to prev range
- valueTo = moment(picker.getValuePickerFrom());
- valueFrom = moment(picker.getValuePickerFrom()).subtract(currentRange, 'seconds');
+ vm.deviceWithoutData = false;
+ vm.scrollToComments = scrollToComments;
- picker.setValuePickers([valueFrom.toDate(), valueTo.toDate()]);
+ $scope.$on('deviceWithoutData', function(ev, data) {
- } else if(direction === 'right') {
- var today = timeUtils.getToday();
- var currentValueTo = picker.getValuePickerTo();
- if( timeUtils.isSameDay(today, timeUtils.getMillisFromDate(currentValueTo)) ) {
- return;
+ $timeout(function() {
+ vm.device = data.device;
+ vm.deviceWithoutData = true;
+
+ if (data.belongsToUser) {
+ vm.user = 'owner';
+ } else {
+ vm.user = 'visitor';
+ }
+ }, 0);
+
+ });
+
+ function scrollToComments(){
+ location.hash = '';
+ location.hash = '#disqus_thread';
}
+ },
+ controllerAs: 'vm'
+ };
+ }
+})();
- valueFrom = moment(picker.getValuePickerTo());
- valueTo = moment(picker.getValuePickerTo()).add(currentRange, 'seconds');
+(function() {
+ 'use strict';
- picker.setValuePickers([valueFrom.toDate(), valueTo.toDate()]);
+ angular.module('app.components')
+ .directive('loadingBackdrop', loadingBackdrop);
- }
- resetTimeOpts();
- }
+ /**
+ * Backdrop for app initialization and between states
+ *
+ */
+ loadingBackdrop.$inject = [];
+ function loadingBackdrop() {
+ return {
+ templateUrl: 'app/core/animation/backdrop/loadingBackdrop.html',
+ controller: function($scope) {
+ var vm = this;
+ vm.isViewLoading = true;
+ vm.mapStateLoading = false;
- //hide everything but the functions to interact with the pickers
- function initializePicker() {
- var range = {};
- /*jshint camelcase: false*/
- var from_$input = angular.element('#picker_from').pickadate({
- onOpen: function(){
- vm.resetTimeOpts();
- },
- onClose: function(){
- angular.element(document.activeElement).blur();
- },
- container: 'body',
- klass: {
- holder: 'picker__holder picker_container'
- }
- });
- var from_picker = from_$input.pickadate('picker');
+ // listen for app loading event
+ $scope.$on('viewLoading', function() {
+ vm.isViewLoading = true;
+ });
- var to_$input = angular.element('#picker_to').pickadate({
- onOpen: function(){
- vm.resetTimeOpts();
- },
- onClose: function(){
- angular.element(document.activeElement).blur();
+ $scope.$on('viewLoaded', function() {
+ vm.isViewLoading = false;
+ });
+
+ // listen for map state loading event
+ $scope.$on('mapStateLoading', function() {
+ if(vm.isViewLoading) {
+ return;
+ }
+ vm.mapStateLoading = true;
+ });
+
+ $scope.$on('mapStateLoaded', function() {
+ vm.mapStateLoading = false;
+ });
},
- container: 'body',
- klass: {
- holder: 'picker__holder picker_container'
- }
- });
+ controllerAs: 'vm'
+ };
+ }
+})();
- var to_picker = to_$input.pickadate('picker');
+(function() {
+ 'use strict';
- if( from_picker.get('value') ) {
- to_picker.set('min', from_picker.get('select') );
- }
- if( to_picker.get('value') ) {
- from_picker.set('max', to_picker.get('select') );
- }
+ angular.module('app.components')
+ .controller('KitController', KitController);
- from_picker.on('close', function(event) {
- setFromRange(getCalculatedFrom(from_picker.get('value')));
- });
+ KitController.$inject = ['$state','$scope', '$stateParams',
+ 'sensor', 'FullDevice', '$mdDialog', 'belongsToUser',
+ 'timeUtils', 'animation', 'auth',
+ '$timeout', 'alert', '$q', 'device',
+ 'HasSensorDevice', 'geolocation', 'PreviewDevice'];
+ function KitController($state, $scope, $stateParams,
+ sensor, FullDevice, $mdDialog, belongsToUser,
+ timeUtils, animation, auth,
+ $timeout, alert, $q, device,
+ HasSensorDevice, geolocation, PreviewDevice) {
- to_picker.on('close', function(event) {
- setToRange(getCalculatedTo(to_picker.get('value')));
- });
+ var vm = this;
+ var sensorsData = [];
- from_picker.on('set', function(event) {
- if(event.select) {
- to_picker.set('min', getFromRange());
- } else if( 'clear' in event) {
- to_picker.set('min', false);
- }
- });
+ var mainSensorID, compareSensorID;
+ var picker;
+ vm.deviceID = $stateParams.id;
+ vm.battery = {};
+ vm.downloadData = downloadData;
+ vm.geolocate = geolocate;
+ vm.device = undefined;
+ vm.deviceBelongsToUser = belongsToUser;
+ vm.deviceWithoutData = false;
+ vm.legacyApiKey = belongsToUser ?
+ auth.getCurrentUser().data.key :
+ undefined;
+ vm.loadingChart = true;
+ vm.moveChart = moveChart;
+ vm.allowUpdateChart = true;
+ vm.ownerDevices = [];
+ vm.removeDevice = removeDevice;
+ vm.resetTimeOpts = resetTimeOpts;
+ vm.sampleDevices = [];
+ vm.selectedSensor = undefined;
+ vm.selectedSensorData = {};
+ vm.selectedSensorToCompare = undefined;
+ vm.selectedSensorToCompareData = {};
+ vm.sensors = [];
+ vm.chartSensors = [];
+ vm.sensorsToCompare = [];
+ vm.setFromLast = setFromLast;
+ vm.showSensorOnChart = showSensorOnChart;
+ vm.showStore = showStore;
+ vm.slide = slide;
+ vm.showRaw = false;
+ vm.timeOpt = ['60 minutes', 'day' , 'month'];
+ vm.timeOptSelected = timeOptSelected;
+ vm.updateInterval = 15000;
+ vm.hasRaw;
+ vm.sensorNames = {};
- to_picker.on('set', function(event) {
- if(event.select) {
- from_picker.set('max', getToRange());
- } else if( 'clear' in event) {
- from_picker.set('max', false);
- }
- });
+ var focused = true;
- //set to-picker max to today
- to_picker.set('max', getLatestUpdated());
+ // event listener on change of value of main sensor selector
+ $scope.$watch('vm.selectedSensor', function(newVal) {
- function getSevenDaysAgoFromLatestUpdate() {
- var lastTime = moment(vm.device.lastReadingAt.raw);
- return lastTime.subtract(7, 'days').valueOf();
+ // Prevents undisered calls if selected sensor is not yet defined
+ if (!newVal) {
+ return;
}
- function getLatestUpdated() {
- return moment(vm.device.lastReadingAt.raw).toDate();
- }
+ vm.selectedSensorToCompare = undefined;
+ vm.selectedSensorToCompareData = {};
+ vm.chartDataCompare = [];
+ compareSensorID = undefined;
- function getCalculatedFrom(pickerTimeFrom) {
- var from,
- pickerTime;
+ setSensorSideChart();
- pickerTime = moment(pickerTimeFrom, 'D MMMM, YYYY');
- from = pickerTime.startOf('day');
+ vm.sensorsToCompare = getSensorsToCompare();
- return from;
- }
+ $timeout(function() {
+ // TODO: Improvement, change how we set the colors
+ colorSensorCompareName();
+ setSensor({type: 'main', value: newVal});
- function getCalculatedTo(pickerTimeTo) {
- var to,
- pickerTime;
+ if (picker){
+ changeChart([mainSensorID]);
+ }
+ }, 100);
- pickerTime = moment(pickerTimeTo, 'D MMMM, YYYY');
+ });
- to = pickerTime.endOf('day');
- if (moment().diff(to) < 0) {
- var now = moment();
- to = pickerTime.set({
- 'hour' : now.get('hour'),
- 'minute' : now.get('minute'),
- 'second' : now.get('second')
- });
- }
+ // event listener on change of value of compare sensor selector
+ $scope.$watch('vm.selectedSensorToCompare', function(newVal, oldVal) {
+ vm.sensorsToCompare.forEach(function(sensor) {
+ if(sensor.id === newVal) {
+ _.extend(vm.selectedSensorToCompareData, sensor);
+ }
+ });
- return to;
- }
+ $timeout(function() {
+ colorSensorCompareName();
+ setSensor({type: 'compare', value: newVal});
- function updateChart() {
- var sensors = [mainSensorID, compareSensorID];
- sensors = sensors.filter(function(sensor) {
- return sensor;
- });
- changeChart(sensors, {
- from: range.from,
- to: range.to
- });
- }
+ if(oldVal === undefined && newVal === undefined) {
+ return;
+ }
+ changeChart([compareSensorID]);
+ }, 100);
- function setFromRange(from) {
- range.from = from;
- from_picker.set('select', getFromRange());
- updateChart();
- }
+ });
- function setToRange(to) {
- range.to = to;
- to_picker.set('select', getToRange());
- updateChart();
- }
+ $scope.$on('hideChartSpinner', function() {
+ vm.loadingChart = false;
+ });
- function getFromRange() {
- return moment(range.from).toDate();
- }
+ $scope.$on('$destroy', function() {
+ focused = false;
+ $timeout.cancel(vm.updateTimeout);
+ });
- function getToRange() {
- return moment(range.to).toDate();
- }
+ $scope.$on('$viewContentLoaded', function(event){
+ initialize();
+ });
- function setRange(from, to) {
- range.from = from;
- range.to = to;
- from_picker.set('select', getFromRange());
- to_picker.set('select', getToRange());
- updateChart();
- }
+ function initialize() {
+ animation.viewLoaded();
+ updatePeriodically();
+ }
- if(vm.device){
- if(vm.device.systemTags.includes('new')){
- var lastUpdate = getLatestUpdated();
- setRange(timeUtils.getHourBefore(lastUpdate), lastUpdate);
- } else if (timeUtils.isWithin(7, 'days', vm.device.lastReadingAt.raw) || !vm.device.lastReadingAt.raw) {
- //set from-picker to seven days ago and set to-picker to today
- setRange(timeUtils.getSevenDaysAgo(), timeUtils.getToday());
- } else {
- // set from-picker to and set to-picker to today
- setRange(getSevenDaysAgoFromLatestUpdate(), getLatestUpdated());
- }
- }
+ function pollAndUpdate(){
+ vm.updateTimeout = $timeout(function() {
+ updatePeriodically();
+ }, vm.updateInterval);
+ }
- // api to interact with the picker from outside
- return {
- getValuePickerFrom: function() {
- return getFromRange();
- },
- setValuePickerFrom: function(newValue) {
- setFromRange(newValue);
- },
- getValuePickerTo: function() {
- return getToRange();
- },
- setValuePickerTo: function(newValue) {
- setToRange(newValue);
- },
- setValuePickers: function(newValues) {
- var from = newValues[0];
- var to = newValues[1];
- setRange(from, to);
- }
- };
+ function updatePeriodically(){
+ getAndUpdateDevice().then(function(){
+ pollAndUpdate();
+ });
}
- function geolocate() {
- if (navigator.geolocation) {
- navigator.geolocation.getCurrentPosition(function(position){
- if(!position){
- alert.error('Please, allow smartcitizen to geolocate your' +
- 'position so we can find a kit near you.');
- return;
- }
+ function getAndUpdateDevice(){
+ // TODO: Improvement UX Change below to && to avoid constant unhandled error
+ // Through reject is not possible
+ if (vm.deviceID || !isNaN(vm.deviceID)){
+ return device.getDevice(vm.deviceID)
+ .then(function(deviceData) {
+ if (deviceData.is_private) {
+ deviceIsPrivate();
+ }
+ var newDevice = new FullDevice(deviceData);
+ vm.prevDevice = vm.device;
- geolocation.grantHTML5Geolocation();
+ if (vm.prevDevice) {
+ /* Kit already loaded. We are waiting for updates */
+ if (vm.prevDevice.state.name !== 'has published' && newDevice.state.name === 'has published'){
+ /* The kit has just published data for the first time. Fully reload the view */
+ return $q.reject({justPublished: true});
+ } else if(new Date(vm.prevDevice.lastReadingAt.raw) >= new Date(newDevice.lastReadingAt.raw)) {
+ /* Break if there's no new data*/
+ return $q.reject();
+ }
+ }
- var location = {
- lat:position.coords.latitude,
- lng:position.coords.longitude
- };
- device.getDevices(location)
- .then(function(data){
- data = data.plain();
+ vm.device = newDevice;
+ setOwnerSampleDevices();
- _(data)
- .chain()
- .map(function(device) {
- return new HasSensorDevice(device);
- })
- .filter(function(device) {
- return !!device.longitude && !!device.latitude;
- })
- .find(function(device) {
- return _.includes(device.labels, 'online');
- })
- .tap(function(closestDevice) {
- if(focused){
- if(closestDevice) {
- $state.go('layout.home.kit', {id: closestDevice.id});
- } else {
- $state.go('layout.home.kit', {id: data[0].id});
- }
- }
- })
- .value();
- });
- });
+ if (vm.device.state.name === 'has published') {
+ /* Device has data */
+ setDeviceOnMap();
+ setChartTimeRange();
+ deviceAnnouncements();
+
+ /*Load sensor if it has already published*/
+ return $q.all([getMainSensors(vm.device), getCompareSensors(vm.device)]);
+ } else {
+ /* Device just loaded and has no data yet */
+ return $q.reject({noSensorData: true});
+ }
+ })
+ .then(setSensors, killSensorsLoading);
}
}
- function downloadData(device){
- $mdDialog.show({
- hasBackdrop: true,
- controller: 'DownloadModalController',
- controllerAs: 'vm',
- templateUrl: 'app/components/download/downloadModal.html',
- clickOutsideToClose: true,
- locals: {thisDevice: device}
- }).then(function(){
- var alert = $mdDialog.alert()
- .title('SUCCESS')
- .textContent('We are processing your data. Soon you will be notified in your inbox')
- .ariaLabel('')
- .ok('OK!')
- .theme('primary')
- .clickOutsideToClose(true);
-
- $mdDialog.show(alert);
- }).catch(function(err){
- if (!err){
- return;
+ function killSensorsLoading(error){
+ if(error) {
+ if(error.status === 404) {
+ $state.go('layout.404');
}
- var errorAlert = $mdDialog.alert()
- .title('ERROR')
- .textContent('Uh-oh, something went wrong')
- .ariaLabel('')
- .ok('D\'oh')
- .theme('primary')
- .clickOutsideToClose(false);
-
- $mdDialog.show(errorAlert);
- });
+ else if (error.justPublished) {
+ $state.transitionTo($state.current, {reloadMap: true, id: vm.deviceID}, {
+ reload: true, inherit: false, notify: true
+ });
+ }
+ else if (error.noSensorData) {
+ deviceHasNoData();
+ }
+ else if (error.status === 403){
+ deviceIsPrivate();
+ }
+ }
}
- function getMainSensors(deviceData) {
- if(!deviceData) {
- return undefined;
+ function deviceAnnouncements(){
+ if(!timeUtils.isWithin(1, 'months', vm.device.lastReadingAt.raw)) {
+ //TODO: Cosmetic Update the message
+ alert.info.longTime();
}
- return deviceData.getSensors({type: 'main'});
- }
- function getCompareSensors(deviceData) {
- if(!vm.device) {
- return undefined;
+ /* The device has just published data after not publishing for 15min */
+ else if(vm.prevDevice && timeUtils.isDiffMoreThan15min(vm.prevDevice.lastReadingAt.raw, vm.device.lastReadingAt.raw)) {
+ alert.success('Your Kit just published again!');
}
- deviceData.getSensors({type: 'compare'});
}
- function getOwnerDevices(deviceData, sampling) {
- if(!deviceData) {
- return undefined;
+
+ function deviceHasNoData() {
+ vm.deviceWithoutData = true;
+ animation.deviceWithoutData({device: vm.device, belongsToUser:vm.deviceBelongsToUser});
+ if(vm.deviceBelongsToUser) {
+ alert.info.noData.owner($stateParams.id);
+ } else {
+ alert.info.noData.visitor();
}
- var deviceIDs = deviceData.owner.devices.slice(sampling);
- // var ownerID = deviceData.owner.id;
- // TODO: Refactor This is in the user endpoint, no need to query devices
- return $q.all(
- deviceIDs.map(function(id) {
- return device.getDevice(id)
- .then(function(data) {
- return new PreviewDevice(data);
- });
- })
- );
}
- function setFromLast(what){
- /* This will not show the last 60 minutes or 24 hours,
- instead it will show the last hour or day*/
- var to, from;
- if (what === '60 minutes') {
- to = moment(vm.device.lastReadingAt.raw);
- from = moment(vm.device.lastReadingAt.raw).subtract(60, 'minutes');
- } else {
- to = moment(vm.device.lastReadingAt.raw).endOf(what);
- from = moment(vm.device.lastReadingAt.raw).startOf(what);
- }
- // Check if we are in the future
- if (moment().diff(to) < 0){
- to = moment(vm.device.lastReadingAt.raw);
- }
- picker.setValuePickers([from.toDate(), to.toDate()]);
+ function deviceIsPrivate() {
+ alert.info.noData.private();
}
- function timeOptSelected(){
- vm.allowUpdateChart = false;
- if (vm.dropDownSelection){
- setFromLast(vm.dropDownSelection);
- }
- }
- function resetTimeOpts(){
- vm.allowUpdateChart = false;
- vm.dropDownSelection = undefined;
+ function setOwnerSampleDevices() {
+ // TODO: Refactor - this information is in the user, no need to go to devices
+ getOwnerDevices(vm.device, -6)
+ .then(function(ownerDevices){
+ vm.sampleDevices = ownerDevices;
+ });
}
- function showStore() {
- $mdDialog.show({
- hasBackdrop: true,
- controller: 'StoreModalController',
- templateUrl: 'app/components/store/storeModal.html',
- clickOutsideToClose: true
- });
+ function setChartTimeRange() {
+ if(vm.allowUpdateChart) {
+ /* Init the chart range to default if doesn't exist of the user hasn't interacted */
+ picker = initializePicker();
+ }
}
- }
-})();
-
-(function() {
- 'use strict';
- angular.module('app.components')
- .controller('NewKitController', NewKitController);
+ function setDeviceOnMap() {
+ animation.deviceLoaded({lat: vm.device.latitude, lng: vm.device.longitude,
+ id: vm.device.id});
+ }
- NewKitController.$inject = ['$scope', '$state', 'animation', 'device', 'tag', 'alert', 'auth', '$stateParams', '$timeout'];
- function NewKitController($scope, $state, animation, device, tag, alert, auth, $stateParams, $timeout) {
- var vm = this;
+ function setSensors(sensorsRes){
- vm.step = 1;
- vm.submitStepOne = submitStepOne;
- vm.backToProfile = backToProfile;
+ var mainSensors = sensorsRes[0];
+ var compareSensors = sensorsRes[1];
- // FORM INFO
- vm.deviceForm = {
- name: undefined,
- exposure: undefined,
- location: {
- lat: undefined,
- lng: undefined,
- zoom: 16
- },
- is_private: false,
- legacyVersion: '1.1',
- tags: []
- };
+ vm.battery = _.find(mainSensors, {name: 'battery'});
+ vm.sensors = mainSensors.reverse();
+ vm.sensors.forEach(checkRaw);
+ vm.sensors.forEach(getHardwareName);
- // EXPOSURE SELECT
- vm.exposure = [
- {name: 'indoor', value: 1},
- {name: 'outdoor', value: 2}
- ];
+ setSensorSideChart();
- // VERSION SELECT
- vm.version = [
- {name: 'Smart Citizen Kit 1.0', value: '1.0'},
- {name: 'Smart Citizen Kit 1.1', value: '1.1'}
- ];
+ if (!vm.selectedSensor) {
+ vm.chartSensors = vm.sensors;
+ vm.sensorsToCompare = compareSensors;
+ vm.selectedSensor = (vm.sensors && vm.sensors[0]) ? vm.sensors[0].id : undefined;
+ }
- $scope.$on('leafletDirectiveMarker.dragend', function(event, args){
- vm.deviceForm.location.lat = args.model.lat;
- vm.deviceForm.location.lng = args.model.lng;
- });
+ animation.mapStateLoaded();
+ }
- // TAGS SELECT
- vm.tags = [];
- $scope.$watch('vm.tag', function(newVal, oldVal) {
- if(!newVal) {
- return;
- }
- // remove selected tag from select element
- vm.tag = undefined;
+ function checkRaw(value){
+ vm.hasRaw |= (value.tags.indexOf('raw') !== -1);
+ }
- var alreadyPushed = _.some(vm.deviceForm.tags, function(tag) {
- return tag.id === newVal;
+ function getHardwareName(value) {
+ vm.sensorNames[value.id] = vm.device.sensors.find(element => element.id === value.id).name;
+ }
+ function setSensorSideChart() {
+ if(vm.sensors){
+ vm.sensors.forEach(function(sensor) {
+ if(sensor.id === vm.selectedSensor) {
+ _.extend(vm.selectedSensorData, sensor);
+ }
});
- if(alreadyPushed) {
- return;
- }
+ }
+ }
- var tag = _.find(vm.tags, function(tag) {
- return tag.id === newVal;
- });
- vm.deviceForm.tags.push(tag);
- });
- vm.removeTag = removeTag;
+ function removeDevice() {
+ var confirm = $mdDialog.confirm()
+ .title('Delete this kit?')
+ .textContent('Are you sure you want to delete this kit?')
+ .ariaLabel('')
+ .ok('DELETE')
+ .cancel('Cancel')
+ .theme('primary')
+ .clickOutsideToClose(true);
- // MAP CONFIGURATION
- var mapBoxToken = 'pk.eyJ1IjoidG9tYXNkaWV6IiwiYSI6ImRTd01HSGsifQ.loQdtLNQ8GJkJl2LUzzxVg';
+ $mdDialog
+ .show(confirm)
+ .then(function(){
+ device
+ .removeDevice(vm.device.id)
+ .then(function(){
+ alert.success('Your kit was deleted successfully');
+ device.updateContext().then(function(){
+ $state.transitionTo('layout.myProfile.kits', $stateParams,
+ { reload: false,
+ inherit: false,
+ notify: true
+ });
+ });
+ })
+ .catch(function(){
+ alert.error('Error trying to delete your kit.');
+ });
+ });
+ }
- vm.getLocation = getLocation;
- vm.markers = {
- main: {
- lat: undefined,
- lng: undefined,
- draggable: true
- }
- };
- vm.tiles = {
- url: 'https://api.mapbox.com/styles/v1/mapbox/streets-v10/tiles/{z}/{x}/{y}?access_token=' + mapBoxToken
- };
- vm.defaults = {
- scrollWheelZoom: false
- };
+ function showSensorOnChart(sensorID) {
+ vm.selectedSensor = sensorID;
+ }
- vm.macAddress = undefined;
+ function slide(direction) {
+ var slideContainer = angular.element('.sensors_container');
+ var scrollPosition = slideContainer.scrollLeft();
+ var width = slideContainer.width();
+ var slideStep = width/2;
- initialize();
+ if(direction === 'left') {
+ slideContainer.animate({'scrollLeft': scrollPosition + slideStep},
+ {duration: 250, queue:false});
+ } else if(direction === 'right') {
+ slideContainer.animate({'scrollLeft': scrollPosition - slideStep},
+ {duration: 250, queue:false});
+ }
+ }
- //////////////
+ function getSensorsToCompare() {
+ return vm.sensors ? vm.sensors.filter(function(sensor) {
+ return sensor.id !== vm.selectedSensor;
+ }) : [];
+ }
- function initialize() {
- animation.viewLoaded();
- getTags();
- vm.userRole = auth.getCurrentUser().data.role;
+ function changeChart(sensorsID, options) {
+ if(!sensorsID[0]) {
+ return;
}
- function getLocation() {
- window.navigator.geolocation.getCurrentPosition(function(position) {
- $scope.$apply(function() {
- var lat = position.coords.latitude;
- var lng = position.coords.longitude;
- vm.deviceForm.location.lat = lat;
- vm.deviceForm.location.lng = lng;
- vm.markers.main.lat = lat;
- vm.markers.main.lng = lng;
- });
- });
+ if(!options) {
+ options = {};
}
+ options.from = options && options.from || picker.getValuePickerFrom();
+ options.to = options && options.to || picker.getValuePickerTo();
- function removeTag(tagID) {
- vm.deviceForm.tags = _.filter(vm.deviceForm.tags, function(tag) {
- return tag.id !== tagID;
+ //show spinner
+ vm.loadingChart = true;
+ //grab chart data and save it
+
+ // it can be either 2 sensors or 1 sensor, so we use $q.all to wait for all
+ $q.all(
+ _.map(sensorsID, function(sensorID) {
+ return getChartData($stateParams.id, sensorID, options.from, options.to)
+ .then(function(data) {
+ return data;
+ });
+ })
+ ).then(function() {
+ // after all sensors resolve, prepare data and attach it to scope
+ // the variable on the scope will pass the data to the chart directive
+ vm.chartDataMain = prepareChartData([mainSensorID, compareSensorID]);
+ });
+ }
+ // calls api to get sensor data and saves it to sensorsData array
+ function getChartData(deviceID, sensorID, dateFrom, dateTo, options) {
+ return sensor.getSensorsData(deviceID, sensorID, dateFrom, dateTo)
+ .then(function(data) {
+ //save sensor data of this kit so that it can be reused
+ sensorsData[sensorID] = data.readings;
+ return data;
});
- }
+ }
- function submitStepOne() {
- var data = {
- name: vm.deviceForm.name,
- description: vm.deviceForm.description,
- exposure: findExposure(vm.deviceForm.exposure),
- latitude: vm.deviceForm.location.lat,
- longitude: vm.deviceForm.location.lng,
- is_private: vm.deviceForm.is_private,
- hardware_version_override: vm.deviceForm.legacyVersion,
- /*jshint camelcase: false */
- user_tags: _.map(vm.deviceForm.tags, 'name').join(',')
- };
+ function prepareChartData(sensorsID) {
+ var compareSensor;
+ var parsedDataMain = parseSensorData(sensorsData, sensorsID[0]);
+ var mainSensor = {
+ data: parsedDataMain,
+ color: vm.selectedSensorData.color,
+ unit: vm.selectedSensorData.unit
+ };
+ if(sensorsID[1] && sensorsID[1] !== -1) {
+ var parsedDataCompare = parseSensorData(sensorsData, sensorsID[1]);
- device.createDevice(data)
- .then(
- function(response) {
- device.updateContext().then(function(){
- auth.setCurrentUser('appLoad').then(function(){
- $timeout($state.go('layout.kitEdit', {id:response.id, step:2}), 2000);
- });
- });
- },
- function(err) {
- vm.errors = err.data.errors;
- alert.error('There has been an error during kit set up');
- });
+ compareSensor = {
+ data: parsedDataCompare,
+ color: vm.selectedSensorToCompareData.color,
+ unit: vm.selectedSensorToCompareData.unit
+ };
}
+ var newChartData = [mainSensor, compareSensor];
+ return newChartData;
+ }
- function getTags() {
- tag.getTags()
- .then(function(tagsData) {
- vm.tags = tagsData;
- });
+ function parseSensorData(data, sensorID) {
+ if(data.length === 0) {
+ return [];
}
+ return data[sensorID].map(function(dataPoint) {
+ var time = timeUtils.formatDate(dataPoint[0]);
+ var value = dataPoint[1];
+ var count = value === null ? 0 : value;
+ return {
+ time: time,
+ count: count,
+ value: value
+ };
+ });
+ }
- function toProfile(){
- $state.transitionTo('layout.myProfile.kits', $stateParams,
- { reload: false,
- inherit: false,
- notify: true
- });
+ function setSensor(options) {
+ var sensorID = options.value;
+ if(sensorID === undefined) {
+ return;
}
-
- function backToProfile(){
- // TODO: Refactor Check
- toProfile();
+ if(options.type === 'main') {
+ mainSensorID = sensorID;
+ } else if(options.type === 'compare') {
+ compareSensorID = sensorID;
}
+ }
- //TODO: move to utils
- function findExposure(nameOrValue) {
- var findProp, resultProp;
- //if it's a string
- if(isNaN(parseInt(nameOrValue))) {
- findProp = 'name';
- resultProp = 'value';
- } else {
- findProp = 'value';
- resultProp = 'name';
- }
-
- var option = _.find(vm.exposure, function(exposureFromList) {
- return exposureFromList[findProp] === nameOrValue;
- });
- if(option) {
- return option[resultProp];
- }
- }
+ function colorSensorCompareName() {
+ var name = angular.element('.sensor_compare').find('md-select-label').find('span');
+ name.css('color', vm.selectedSensorToCompareData.color || 'white');
+ var icon = angular.element('.sensor_compare').find('md-select-label').find('.md-select-icon');
+ icon.css('color', 'white');
}
-})();
-(function() {
- 'use strict';
+ function getCurrentRange() {
+ var to = moment(picker.getValuePickerTo());
+ var from = moment(picker.getValuePickerFrom());
+ return to.diff(from)/1000;
+ }
- // Taken from this answer on SO:
- // https://stackoverflow.com/questions/17893708/angularjs-textarea-bind-to-json-object-shows-object-object
- angular.module('app.components').directive('jsonText', function() {
- return {
- restrict: 'A',
- require: 'ngModel',
- link: function(scope, element, attr, ngModel){
- function into(input) {
- return JSON.parse(input);
- }
- function out(data) {
- return JSON.stringify(data);
- }
- ngModel.$parsers.push(into);
- ngModel.$formatters.push(out);
- }
- };
- });
+ function moveChart(direction) {
- angular.module('app.components')
- .controller('EditKitController', EditKitController);
+ var valueTo, valueFrom;
+ //grab current date range
+ var currentRange = getCurrentRange();
- EditKitController.$inject = ['$scope', '$element', '$location', '$timeout', '$state',
- 'animation','auth','device', 'tag', 'alert', 'step', '$stateParams', 'FullDevice'];
- function EditKitController($scope, $element, $location, $timeout, $state, animation,
- auth, device, tag, alert, step, $stateParams, FullDevice) {
+ /*jshint camelcase: false*/
+ var from_picker = angular.element('#picker_from').pickadate('picker');
+ var to_picker = angular.element('#picker_to').pickadate('picker');
- var vm = this;
+ if(direction === 'left') {
+ //set both from and to pickers to prev range
+ valueTo = moment(picker.getValuePickerFrom());
+ valueFrom = moment(picker.getValuePickerFrom()).subtract(currentRange, 'seconds');
- // WAIT INTERVAL FOR USER FEEDBACK and TRANSITIONS (This will need to change)
- var timewait = {
- long: 5000,
- normal: 2000,
- short: 1000
- };
+ picker.setValuePickers([valueFrom.toDate(), valueTo.toDate()]);
- vm.step = step;
+ } else if(direction === 'right') {
+ var today = timeUtils.getToday();
+ var currentValueTo = picker.getValuePickerTo();
+ if( timeUtils.isSameDay(today, timeUtils.getMillisFromDate(currentValueTo)) ) {
+ return;
+ }
- // KEY USER ACTIONS
- vm.submitFormAndKit = submitFormAndKit;
- vm.backToProfile = backToProfile;
- vm.backToDevice = backToDevice;
- vm.submitForm = submitForm;
- vm.goToStep = goToStep;
- vm.nextAction = 'save';
+ valueFrom = moment(picker.getValuePickerTo());
+ valueTo = moment(picker.getValuePickerTo()).add(currentRange, 'seconds');
- // EXPOSURE SELECT
- vm.exposure = [
- {name: 'indoor', value: 1},
- {name: 'outdoor', value: 2}
- ];
+ picker.setValuePickers([valueFrom.toDate(), valueTo.toDate()]);
- // FORM INFO
- vm.deviceForm = {};
- vm.device = undefined;
+ }
+ resetTimeOpts();
+ }
- $scope.clearSearchTerm = function() {
- $scope.searchTerm = '';
- };
- // The md-select directive eats keydown events for some quick select
- // logic. Since we have a search input here, we don't need that logic.
- $element.find('input').on('keydown', function(ev) {
- ev.stopPropagation();
+ //hide everything but the functions to interact with the pickers
+ function initializePicker() {
+ var range = {};
+ /*jshint camelcase: false*/
+ var from_$input = angular.element('#picker_from').pickadate({
+ onOpen: function(){
+ vm.resetTimeOpts();
+ },
+ onClose: function(){
+ angular.element(document.activeElement).blur();
+ },
+ container: 'body',
+ klass: {
+ holder: 'picker__holder picker_container'
+ }
});
+ var from_picker = from_$input.pickadate('picker');
- $scope.$on('leafletDirectiveMarker.dragend', function(event, args){
- vm.deviceForm.location.lat = args.model.lat;
- vm.deviceForm.location.lng = args.model.lng;
+ var to_$input = angular.element('#picker_to').pickadate({
+ onOpen: function(){
+ vm.resetTimeOpts();
+ },
+ onClose: function(){
+ angular.element(document.activeElement).blur();
+ },
+ container: 'body',
+ klass: {
+ holder: 'picker__holder picker_container'
+ }
});
- // MAP CONFIGURATION
- var mapBoxToken = 'pk.eyJ1IjoidG9tYXNkaWV6IiwiYSI6ImRTd01HSGsifQ.loQdtLNQ8GJkJl2LUzzxVg';
+ var to_picker = to_$input.pickadate('picker');
- vm.getLocation = getLocation;
- vm.markers = {};
- vm.tiles = {
- url: 'https://api.mapbox.com/styles/v1/mapbox/streets-v10/tiles/{z}/{x}/{y}?access_token=' + mapBoxToken
- };
- vm.defaults = {
- scrollWheelZoom: false
- };
+ if( from_picker.get('value') ) {
+ to_picker.set('min', from_picker.get('select') );
+ }
+ if( to_picker.get('value') ) {
+ from_picker.set('max', to_picker.get('select') );
+ }
- initialize();
+ from_picker.on('close', function(event) {
+ setFromRange(getCalculatedFrom(from_picker.get('value')));
+ });
- /////////////////
+ to_picker.on('close', function(event) {
+ setToRange(getCalculatedTo(to_picker.get('value')));
+ });
- function initialize() {
- var deviceID = $stateParams.id;
+ from_picker.on('set', function(event) {
+ if(event.select) {
+ to_picker.set('min', getFromRange());
+ } else if( 'clear' in event) {
+ to_picker.set('min', false);
+ }
+ });
- animation.viewLoaded();
- getTags();
-
- if (!deviceID || deviceID === ''){
- return;
+ to_picker.on('set', function(event) {
+ if(event.select) {
+ from_picker.set('max', getToRange());
+ } else if( 'clear' in event) {
+ from_picker.set('max', false);
}
- device.getDevice(deviceID)
- .then(function(deviceData) {
- vm.device = new FullDevice(deviceData);
- vm.userRole = auth.getCurrentUser().data.role;
- vm.deviceForm = {
- name: vm.device.name,
- exposure: findExposureFromLabels(vm.device.systemTags),
- location: {
- lat: vm.device.location.latitude,
- lng: vm.device.location.longitude,
- zoom: 16
- },
- is_private: vm.device.isPrivate,
- precise_location: vm.device.preciseLocation,
- enable_forwarding: vm.device.enableForwarding,
- notify_low_battery: vm.device.notifications.lowBattery,
- notify_stopped_publishing: vm.device.notifications.stopPublishing,
- tags: vm.device.userTags,
- postprocessing: vm.device.postProcessing,
- description: vm.device.description,
- hardwareName: vm.device.hardware.name
- };
- vm.markers = {
- main: {
- lat: vm.device.location.latitude,
- lng: vm.device.location.longitude,
- draggable: true
- }
- };
-
- if (vm.device.isLegacy) {
- vm.deviceForm.macAddress = vm.device.macAddress;
- }
- });
- }
+ });
- // Return tags in a comma separated list
- function joinSelectedTags(){
- let tmp = []
- $scope.selectedTags.forEach(function(e){
- tmp.push(e.name)
- })
- return tmp.join(', ');
- }
+ //set to-picker max to today
+ to_picker.set('max', getLatestUpdated());
- function getLocation() {
- window.navigator.geolocation.getCurrentPosition(function(position) {
- $scope.$apply(function() {
- var lat = position.coords.latitude;
- var lng = position.coords.longitude;
- vm.deviceForm.location.lat = lat;
- vm.deviceForm.location.lng = lng;
- vm.markers.main.lat = lat;
- vm.markers.main.lng = lng;
- });
- });
+ function getSevenDaysAgoFromLatestUpdate() {
+ var lastTime = moment(vm.device.lastReadingAt.raw);
+ return lastTime.subtract(7, 'days').valueOf();
}
- function submitFormAndKit(){
- submitForm(backToProfile, timewait.normal);
+ function getLatestUpdated() {
+ return moment(vm.device.lastReadingAt.raw).toDate();
}
- function submitForm(next, delayTransition) {
- var data = {
- name: vm.deviceForm.name,
- description: vm.deviceForm.description,
- postprocessing_attributes: vm.deviceForm.postprocessing,
- exposure: findExposure(vm.deviceForm.exposure),
- latitude: vm.deviceForm.location.lat,
- longitude: vm.deviceForm.location.lng,
- is_private: vm.deviceForm.is_private,
- enable_forwarding: vm.deviceForm.enable_forwarding,
- precise_location: vm.deviceForm.precise_location,
- notify_low_battery: vm.deviceForm.notify_low_battery,
- notify_stopped_publishing: vm.deviceForm.notify_stopped_publishing,
- mac_address: "",
- /*jshint camelcase: false */
- user_tags: joinSelectedTags(),
- };
-
- vm.errors={};
+ function getCalculatedFrom(pickerTimeFrom) {
+ var from,
+ pickerTime;
- if(!vm.device.isSCK) {
- data.hardware_name_override = vm.deviceForm.hardwareName;
- }
+ pickerTime = moment(pickerTimeFrom, 'D MMMM, YYYY');
+ from = pickerTime.startOf('day');
- // Workaround for the mac_address bypass
- // If mac_address is "", we get an error on the request -> we use it for the newKit
- // If mac_address is null, no problem -> we use it for the
- if ($stateParams.step === "2") {
- data.mac_address = vm.deviceForm.macAddress ? vm.deviceForm.macAddress : "";
- } else {
- data.mac_address = vm.deviceForm.macAddress ? vm.deviceForm.macAddress : null;
- }
+ return from;
+ }
- device.updateDevice(vm.device.id, data)
- .then(
- function() {
+ function getCalculatedTo(pickerTimeTo) {
+ var to,
+ pickerTime;
- if (next){
- alert.success('Your kit was updated!');
- }
+ pickerTime = moment(pickerTimeTo, 'D MMMM, YYYY');
- device.updateContext().then(function(){
- if (next){
- $timeout(next, delayTransition);
- }
- });
- })
- .catch(function(err) {
- if(err.data.errors) {
- vm.errors = err.data.errors;
- var message = Object.keys(vm.errors).map(function (key, _) {
- return [key, vm.errors[key][0]].join(' '); }).join('');
- alert.error('Oups! Check the input. Something went wrong!');
- throw new Error('[Client:error] ' + message);
- }
- $timeout(function(){ }, timewait.long);
+ to = pickerTime.endOf('day');
+ if (moment().diff(to) < 0) {
+ var now = moment();
+ to = pickerTime.set({
+ 'hour' : now.get('hour'),
+ 'minute' : now.get('minute'),
+ 'second' : now.get('second')
});
- }
+ }
- function findExposureFromLabels(labels){
- var label = vm.exposure.filter(function(n) {
- return labels.indexOf(n.name) !== -1;
- })[0];
- if(label) {
- return findExposure(label.name);
- } else {
- return findExposure(vm.exposure[0].name);
- }
+ return to;
}
- function findExposure(nameOrValue) {
- var findProp, resultProp;
-
- //if it's a string
- if(isNaN(parseInt(nameOrValue))) {
- findProp = 'name';
- resultProp = 'value';
- } else {
- findProp = 'value';
- resultProp = 'name';
- }
-
- var option = _.find(vm.exposure, function(exposureFromList) {
- return exposureFromList[findProp] === nameOrValue;
+ function updateChart() {
+ var sensors = [mainSensorID, compareSensorID];
+ sensors = sensors.filter(function(sensor) {
+ return sensor;
+ });
+ changeChart(sensors, {
+ from: range.from,
+ to: range.to
});
- if(option) {
- return option[resultProp];
- } else {
- return vm.exposure[0][resultProp];
- }
}
- function getTags() {
- tag.getTags()
- .then(function(tagsData) {
- vm.tags = tagsData;
- });
+ function setFromRange(from) {
+ range.from = from;
+ from_picker.set('select', getFromRange());
+ updateChart();
}
- function backToProfile(){
- $state.transitionTo('layout.myProfile.kits', $stateParams,
- { reload: false,
- inherit: false,
- notify: true
- });
+ function setToRange(to) {
+ range.to = to;
+ to_picker.set('select', getToRange());
+ updateChart();
}
- function backToDevice(){
- $state.transitionTo('layout.home.kit', $stateParams,
- { reload: false,
- inherit: false,
- notify: true
- });
+ function getFromRange() {
+ return moment(range.from).toDate();
}
- function goToStep(step) {
- vm.step = step;
- $state.transitionTo('layout.kitEdit', { id:$stateParams.id, step: step} ,
- {
- reload: false,
- inherit: false,
- notify: false
- });
+ function getToRange() {
+ return moment(range.to).toDate();
}
- }
-})();
-
-(function() {
- 'use strict';
-
- angular.module('app.components')
- .factory('User', ['COUNTRY_CODES', function(COUNTRY_CODES) {
-
- /**
- * User constructor
- * @param {Object} userData - User data sent from API
- * @property {number} id - User ID
- * @property {string} username - Username
- * @property {string} profile_picture - Avatar URL of user
- * @property {Array} devices - Kits that belongs to this user
- * @property {string} url - URL
- * @property {string} city - User city
- * @property {string} country - User country
- */
- function User(userData) {
- this.id = userData.id;
- this.username = userData.username;
- this.profile_picture = userData.profile_picture;
- this.devices = userData.devices;
- this.url = userData.url;
- this.city = userData.location.city;
- /*jshint camelcase: false */
- this.country = COUNTRY_CODES[userData.location.country_code];
+ function setRange(from, to) {
+ range.from = from;
+ range.to = to;
+ from_picker.set('select', getFromRange());
+ to_picker.set('select', getToRange());
+ updateChart();
}
- return User;
- }]);
-})();
-
-(function() {
- 'use strict';
-
- angular.module('app.components')
- .factory('NonAuthUser', ['User', function(User) {
-
- function NonAuthUser(userData) {
- User.call(this, userData);
+ if(vm.device){
+ if(vm.device.systemTags.includes('new')){
+ var lastUpdate = getLatestUpdated();
+ setRange(timeUtils.getHourBefore(lastUpdate), lastUpdate);
+ } else if (timeUtils.isWithin(7, 'days', vm.device.lastReadingAt.raw) || !vm.device.lastReadingAt.raw) {
+ //set from-picker to seven days ago and set to-picker to today
+ setRange(timeUtils.getSevenDaysAgo(), timeUtils.getToday());
+ } else {
+ // set from-picker to and set to-picker to today
+ setRange(getSevenDaysAgoFromLatestUpdate(), getLatestUpdated());
+ }
}
- NonAuthUser.prototype = Object.create(User.prototype);
- NonAuthUser.prototype.constructor = User;
-
- return NonAuthUser;
- }]);
-})();
-(function() {
- 'use strict';
+ // api to interact with the picker from outside
+ return {
+ getValuePickerFrom: function() {
+ return getFromRange();
+ },
+ setValuePickerFrom: function(newValue) {
+ setFromRange(newValue);
+ },
+ getValuePickerTo: function() {
+ return getToRange();
+ },
+ setValuePickerTo: function(newValue) {
+ setToRange(newValue);
+ },
+ setValuePickers: function(newValues) {
+ var from = newValues[0];
+ var to = newValues[1];
+ setRange(from, to);
+ }
+ };
+ }
- angular.module('app.components')
- .factory('AuthUser', ['User', function(User) {
+ function geolocate() {
+ if (navigator.geolocation) {
+ navigator.geolocation.getCurrentPosition(function(position){
+ if(!position){
+ alert.error('Please, allow smartcitizen to geolocate your' +
+ 'position so we can find a kit near you.');
+ return;
+ }
- /**
- * AuthUser constructor. Used for authenticated users
- * @extends User
- * @param {Object} userData - Contains user data sent from API
- * @property {string} email - User email
- * @property {string} role - User role. Ex: admin
- * @property {string} key - Personal API Key
- */
+ geolocation.grantHTML5Geolocation();
- function AuthUser(userData) {
- User.call(this, userData);
+ var location = {
+ lat:position.coords.latitude,
+ lng:position.coords.longitude
+ };
+ device.getDevices(location)
+ .then(function(data){
+ data = data.plain();
- this.email = userData.email;
- this.role = userData.role;
- /*jshint camelcase: false */
- this.key = userData.legacy_api_key;
+ _(data)
+ .chain()
+ .map(function(device) {
+ return new HasSensorDevice(device);
+ })
+ .filter(function(device) {
+ return !!device.longitude && !!device.latitude;
+ })
+ .find(function(device) {
+ return _.includes(device.labels, 'online');
+ })
+ .tap(function(closestDevice) {
+ if(focused){
+ if(closestDevice) {
+ $state.go('layout.home.kit', {id: closestDevice.id});
+ } else {
+ $state.go('layout.home.kit', {id: data[0].id});
+ }
+ }
+ })
+ .value();
+ });
+ });
}
- AuthUser.prototype = Object.create(User.prototype);
- AuthUser.prototype.constructor = User;
-
- return AuthUser;
- }]);
-})();
+ }
-(function() {
- 'use strict';
+ function downloadData(device){
+ $mdDialog.show({
+ hasBackdrop: true,
+ controller: 'DownloadModalController',
+ controllerAs: 'vm',
+ templateUrl: 'app/components/download/downloadModal.html',
+ clickOutsideToClose: true,
+ locals: {thisDevice: device}
+ }).then(function(){
+ var alert = $mdDialog.alert()
+ .title('SUCCESS')
+ .textContent('We are processing your data. Soon you will be notified in your inbox')
+ .ariaLabel('')
+ .ok('OK!')
+ .theme('primary')
+ .clickOutsideToClose(true);
- angular.module('app.components')
- .factory('Sensor', ['sensorUtils', 'timeUtils', function(sensorUtils, timeUtils) {
+ $mdDialog.show(alert);
+ }).catch(function(err){
+ if (!err){
+ return;
+ }
+ var errorAlert = $mdDialog.alert()
+ .title('ERROR')
+ .textContent('Uh-oh, something went wrong')
+ .ariaLabel('')
+ .ok('D\'oh')
+ .theme('primary')
+ .clickOutsideToClose(false);
- /**
- * Sensor constructor
- * @param {Object} sensorData - Contains the data of a sensor sent from the API
- * @property {string} name - Name of sensor
- * @property {number} id - ID of sensor
- * @property {string} unit - Unit of sensor. Ex: %
- * @property {string} value - Last value sent. Ex: 95
- * @property {string} prevValue - Previous value before last value
- * @property {string} lastReadingAt - last_reading_at for the sensor reading
- * @property {string} icon - Icon URL for sensor
- * @property {string} arrow - Icon URL for sensor trend(up, down or equal)
- * @property {string} color - Color that belongs to sensor
- * @property {object} measurement - Measurement
- * @property {string} fullDescription - Full Description for popup
- * @property {string} previewDescription - Short Description for dashboard. Max 140 chars
- * @property {string} tags - Contains sensor tags for filtering the view
- */
- function Sensor(sensorData) {
+ $mdDialog.show(errorAlert);
+ });
+ }
- this.id = sensorData.id;
- this.name = sensorData.name;
- this.unit = sensorData.unit;
- this.value = sensorUtils.getSensorValue(sensorData);
- this.prevValue = sensorUtils.getSensorPrevValue(sensorData);
- this.lastReadingAt = timeUtils.parseDate(sensorData.last_reading_at);
- this.icon = sensorUtils.getSensorIcon(this.name);
- this.arrow = sensorUtils.getSensorArrow(this.value, this.prevValue);
- this.color = sensorUtils.getSensorColor(this.name);
- this.measurement = sensorData.measurement;
+ function getMainSensors(deviceData) {
+ if(!deviceData) {
+ return undefined;
+ }
+ return deviceData.getSensors({type: 'main'});
+ }
+ function getCompareSensors(deviceData) {
+ if(!vm.device) {
+ return undefined;
+ }
+ deviceData.getSensors({type: 'compare'});
+ }
+ function getOwnerDevices(deviceData, sampling) {
+ if(!deviceData) {
+ return undefined;
+ }
+ var deviceIDs = deviceData.owner.devices.slice(sampling);
+ // var ownerID = deviceData.owner.id;
+ // TODO: Refactor This is in the user endpoint, no need to query devices
+ return $q.all(
+ deviceIDs.map(function(id) {
+ return device.getDevice(id)
+ .then(function(data) {
+ return new PreviewDevice(data);
+ });
+ })
+ );
+ }
- // Some sensors don't have measurements because they are ancestors
- if (sensorData.measurement) {
- var description = sensorData.measurement.description;
- this.fullDescription = description;
- this.previewDescription = description.length > 140 ? description.slice(
- 0, 140).concat(' ... ') : description;
- this.is_ancestor = false;
- } else {
- this.is_ancestor = true;
- }
+ function setFromLast(what){
+ /* This will not show the last 60 minutes or 24 hours,
+ instead it will show the last hour or day*/
+ var to, from;
+ if (what === '60 minutes') {
+ to = moment(vm.device.lastReadingAt.raw);
+ from = moment(vm.device.lastReadingAt.raw).subtract(60, 'minutes');
+ } else {
+ to = moment(vm.device.lastReadingAt.raw).endOf(what);
+ from = moment(vm.device.lastReadingAt.raw).startOf(what);
+ }
+ // Check if we are in the future
+ if (moment().diff(to) < 0){
+ to = moment(vm.device.lastReadingAt.raw);
+ }
+ picker.setValuePickers([from.toDate(), to.toDate()]);
+ }
- // Get sensor tags
- this.tags = sensorData.tags;
+ function timeOptSelected(){
+ vm.allowUpdateChart = false;
+ if (vm.dropDownSelection){
+ setFromLast(vm.dropDownSelection);
}
+ }
+ function resetTimeOpts(){
+ vm.allowUpdateChart = false;
+ vm.dropDownSelection = undefined;
+ }
- return Sensor;
- }]);
+ function showStore() {
+ $mdDialog.show({
+ hasBackdrop: true,
+ controller: 'StoreModalController',
+ templateUrl: 'app/components/store/storeModal.html',
+ clickOutsideToClose: true
+ });
+ }
+ }
})();
+
(function() {
'use strict';
angular.module('app.components')
- .factory('SearchResultLocation', ['SearchResult', function(SearchResult) {
+ .controller('NewKitController', NewKitController);
- /**
- * Search Result Location constructor
- * @extends SearchResult
- * @param {Object} object - Object that contains the search result data from API
- * @property {number} lat - Latitude
- * @property {number} lng - Longitude
- */
- function SearchResultLocation(object) {
- SearchResult.call(this, object);
-
- this.lat = object.latitude;
- this.lng = object.longitude;
- this.layer = object.layer;
- }
- return SearchResultLocation;
- }]);
-
-})();
-
-(function() {
- 'use strict';
+ NewKitController.$inject = ['$scope', '$state', 'animation', 'device', 'tag', 'alert', 'auth', '$stateParams', '$timeout'];
+ function NewKitController($scope, $state, animation, device, tag, alert, auth, $stateParams, $timeout) {
+ var vm = this;
- angular.module('app.components')
- .factory('SearchResult', ['searchUtils', function(searchUtils) {
+ vm.step = 1;
+ vm.submitStepOne = submitStepOne;
+ vm.backToProfile = backToProfile;
- /**
- * Search Result constructor
- * @param {Object} object - Object that belongs to a search result from API
- * @property {string} type - Type of search result. Ex: Country, City, User, Device
- * @property {number} id - ID of search result, only for user & device
- * @property {string} name - Name of search result, only for user & device
- * @property {string} location - Location of search result. Ex: 'Paris, France'
- * @property {string} icon - URL for the icon that belongs to this search result
- * @property {string} iconType - Type of icon. Can be either img or div
- */
-
- function SearchResult(object) {
- this.type = object.type;
- this.id = object.id;
- this.name = searchUtils.parseName(object);
- this.location = searchUtils.parseLocation(object);
- this.icon = searchUtils.parseIcon(object, this.type);
- this.iconType = searchUtils.parseIconType(this.type);
- }
- return SearchResult;
- }]);
-})();
+ // FORM INFO
+ vm.deviceForm = {
+ name: undefined,
+ exposure: undefined,
+ location: {
+ lat: undefined,
+ lng: undefined,
+ zoom: 16
+ },
+ is_private: false,
+ legacyVersion: '1.1',
+ tags: []
+ };
-(function() {
- 'use strict';
+ // EXPOSURE SELECT
+ vm.exposure = [
+ {name: 'indoor', value: 1},
+ {name: 'outdoor', value: 2}
+ ];
- angular.module('app.components')
- .factory('Marker', ['deviceUtils', 'markerUtils', 'timeUtils', '$state', function(deviceUtils, markerUtils, timeUtils, $state) {
- /**
- * Marker constructor
- * @constructor
- * @param {Object} deviceData - Object with data about marker from API
- * @property {number} lat - Latitude
- * @property {number} lng - Longitude
- * @property {string} message - Message inside marker popup
- * @property {Object} icon - Object with classname, size and type of marker icon
- * @property {string} layer - Map layer that icons belongs to
- * @property {boolean} focus - Whether marker popup is opened
- * @property {Object} myData - Marker id and labels
- */
- function Marker(deviceData) {
- let linkStart = '', linkEnd = '';
- const id = deviceData.id;
- if ($state.$current.name === 'embbed') {
- linkStart = '';
- linkEnd = ' ';
- }
- this.lat = deviceUtils.parseCoordinates(deviceData).lat;
- this.lng = deviceUtils.parseCoordinates(deviceData).lng;
- // TODO: Bug, pop-up lastreading at doesn't get updated by publication
- this.message = '';
+ // VERSION SELECT
+ vm.version = [
+ {name: 'Smart Citizen Kit 1.0', value: '1.0'},
+ {name: 'Smart Citizen Kit 1.1', value: '1.1'}
+ ];
- this.icon = markerUtils.getIcon(deviceData);
- this.layer = 'devices';
- this.focus = false;
- this.myData = {
- id: id,
- labels: deviceUtils.parseSystemTags(deviceData),
- tags: deviceUtils.parseUserTags(deviceData)
- };
- }
- return Marker;
+ $scope.$on('leafletDirectiveMarker.dragend', function(event, args){
+ vm.deviceForm.location.lat = args.model.lat;
+ vm.deviceForm.location.lng = args.model.lng;
+ });
- function createTagsTemplate(tagsArr, tagType, clickable) {
- if(typeof(clickable) === 'undefined'){
- clickable = false;
- }
- var clickablTag = '';
- if(clickable){
- clickablTag = 'clickable';
+ // TAGS SELECT
+ vm.tags = [];
+ $scope.$watch('vm.tag', function(newVal, oldVal) {
+ if(!newVal) {
+ return;
}
+ // remove selected tag from select element
+ vm.tag = undefined;
- if(!tagType){
- tagType = 'tag';
+ var alreadyPushed = _.some(vm.deviceForm.tags, function(tag) {
+ return tag.id === newVal;
+ });
+ if(alreadyPushed) {
+ return;
}
- return _.reduce(tagsArr, function(acc, label) {
- var element ='';
- if(tagType === 'tag'){
- element = ' ';
- }else{
- element = ''+label+' ';
- }
- return acc.concat(element);
- }, '');
- }
-
- }]);
-})();
-
-(function () {
- 'use strict';
-
- angular.module('app.components')
- .factory('PreviewDevice', ['Device', function (Device) {
+ var tag = _.find(vm.tags, function(tag) {
+ return tag.id === newVal;
+ });
+ vm.deviceForm.tags.push(tag);
+ });
+ vm.removeTag = removeTag;
- /**
- * Preview Device constructor.
- * Used for devices stacked in a list, like in User Profile or Device states
- * @extends Device
- * @constructor
- * @param {Object} object - Object with all the data about the device from the API
- */
- function PreviewDevice(object) {
- Device.call(this, object);
+ // MAP CONFIGURATION
+ var mapBoxToken = 'pk.eyJ1IjoidG9tYXNkaWV6IiwiYSI6ImRTd01HSGsifQ.loQdtLNQ8GJkJl2LUzzxVg';
- this.dropdownOptions = [];
- this.dropdownOptions.push({ text: 'EDIT', value: '1', href: 'kits/' + this.id + '/edit', icon: 'fa fa-edit' });
- this.dropdownOptions.push({ text: 'SD CARD UPLOAD', value: '2', href: 'kits/' + this.id + '/upload', icon: 'fa fa-sd-card' });
- }
- PreviewDevice.prototype = Object.create(Device.prototype);
- PreviewDevice.prototype.constructor = Device;
- return PreviewDevice;
- }]);
-})();
+ vm.getLocation = getLocation;
+ vm.markers = {
+ main: {
+ lat: undefined,
+ lng: undefined,
+ draggable: true
+ }
+ };
+ vm.tiles = {
+ url: 'https://api.mapbox.com/styles/v1/mapbox/streets-v10/tiles/{z}/{x}/{y}?access_token=' + mapBoxToken
+ };
+ vm.defaults = {
+ scrollWheelZoom: false
+ };
-(function() {
- 'use strict';
+ vm.macAddress = undefined;
- angular.module('app.components')
- .factory('HasSensorDevice', ['Device', function(Device) {
+ initialize();
- function HasSensorDevice(object) {
- Device.call(this, object);
+ //////////////
- this.sensors = object.data.sensors;
- this.longitude = object.data.location.longitude;
- this.latitude = object.data.location.latitude;
+ function initialize() {
+ animation.viewLoaded();
+ getTags();
+ vm.userRole = auth.getCurrentUser().data.role;
}
- HasSensorDevice.prototype = Object.create(Device.prototype);
- HasSensorDevice.prototype.constructor = Device;
-
- HasSensorDevice.prototype.sensorsHasData = function() {
- var parsedSensors = this.sensors.map(function(sensor) {
- return sensor.value;
+ function getLocation() {
+ window.navigator.geolocation.getCurrentPosition(function(position) {
+ $scope.$apply(function() {
+ var lat = position.coords.latitude;
+ var lng = position.coords.longitude;
+ vm.deviceForm.location.lat = lat;
+ vm.deviceForm.location.lng = lng;
+ vm.markers.main.lat = lat;
+ vm.markers.main.lng = lng;
+ });
});
+ }
- return _.some(parsedSensors, function(sensorValue) {
- return !!sensorValue;
+ function removeTag(tagID) {
+ vm.deviceForm.tags = _.filter(vm.deviceForm.tags, function(tag) {
+ return tag.id !== tagID;
});
- };
-
- return HasSensorDevice;
- }]);
-})();
-
-(function() {
- 'use strict';
+ }
- angular.module('app.components')
- .factory('FullDevice', ['Device', 'Sensor', 'deviceUtils', function(Device, Sensor, deviceUtils) {
+ function submitStepOne() {
+ var data = {
+ name: vm.deviceForm.name,
+ description: vm.deviceForm.description,
+ exposure: findExposure(vm.deviceForm.exposure),
+ latitude: vm.deviceForm.location.lat,
+ longitude: vm.deviceForm.location.lng,
+ is_private: vm.deviceForm.is_private,
+ hardware_version_override: vm.deviceForm.legacyVersion,
+ /*jshint camelcase: false */
+ user_tags: _.map(vm.deviceForm.tags, 'name').join(',')
+ };
- /**
- * Full Device constructor.
- * @constructor
- * @extends Device
- * @param {Object} object - Object with all the data about the device from the API
- * @property {Object} owner - Device owner data
- * @property {Array} data - Device sensor's data
- * @property {Array} sensors - Device sensors data
- * @property {Array} postProcessing - Device postprocessing
- */
- function FullDevice(object) {
- Device.call(this, object);
+ device.createDevice(data)
+ .then(
+ function(response) {
+ device.updateContext().then(function(){
+ auth.setCurrentUser('appLoad').then(function(){
+ $timeout($state.go('layout.kitEdit', {id:response.id, step:2}), 2000);
+ });
+ });
+ },
+ function(err) {
+ vm.errors = err.data.errors;
+ alert.error('There has been an error during kit set up');
+ });
+ }
- this.owner = deviceUtils.parseOwner(object);
- this.postProcessing = object.postprocessing;
- this.data = object.data;
- this.sensors = object.data.sensors;
+ function getTags() {
+ tag.getTags()
+ .then(function(tagsData) {
+ vm.tags = tagsData;
+ });
}
- FullDevice.prototype = Object.create(Device.prototype);
- FullDevice.prototype.constructor = FullDevice;
+ function toProfile(){
+ $state.transitionTo('layout.myProfile.kits', $stateParams,
+ { reload: false,
+ inherit: false,
+ notify: true
+ });
+ }
- FullDevice.prototype.getSensors = function(options) {
- var sensors = _(this.data.sensors)
- .chain()
- .map(function(sensor) {
- return new Sensor(sensor);
- }).sort(function(a, b) {
- /* This is a temporary hack to set always PV panel at the end*/
- if (a.id === 18){ return -1;}
- if (b.id === 18){ return 1;}
- /* This is a temporary hack to set always the Battery at the end*/
- if (a.id === 17){ return -1;}
- if (b.id === 17){ return 1;}
- /* This is a temporary hack to set always the Battery at the end*/
- if (a.id === 10){ return -1;}
- if (b.id === 10){ return 1;}
- /* After the hacks, sort the sensors by id */
- return b.id - a.id;
- })
- .tap(function(sensors) {
- if(options.type === 'compare') {
- sensors.unshift({
- name: 'NONE',
- color: 'white',
- id: -1
- });
- }
- })
- .value();
- return sensors;
- };
+ function backToProfile(){
+ // TODO: Refactor Check
+ toProfile();
+ }
- return FullDevice;
- }]);
+ //TODO: move to utils
+ function findExposure(nameOrValue) {
+ var findProp, resultProp;
+ //if it's a string
+ if(isNaN(parseInt(nameOrValue))) {
+ findProp = 'name';
+ resultProp = 'value';
+ } else {
+ findProp = 'value';
+ resultProp = 'name';
+ }
+
+ var option = _.find(vm.exposure, function(exposureFromList) {
+ return exposureFromList[findProp] === nameOrValue;
+ });
+ if(option) {
+ return option[resultProp];
+ }
+ }
+ }
})();
(function() {
'use strict';
- angular.module('app.components')
- .factory('Device', ['deviceUtils', 'timeUtils', function(deviceUtils, timeUtils) {
-
- /**
- * Device constructor.
- * @constructor
- * @param {Object} object - Object with all the data about the device from the API
- * @property {number} id - ID of the device
- * @property {string} name - Name of the device
- * @property {string} state - State of the device. Ex: Never published
- * @property {string} description - Device description
- * @property {Array} systemTags - System tags
- * @property {Array} userTags - User tags. Ex: ''
- * @property {bool} isPrivate - True if private device
- * @property {Array} notifications - Notifications for low battery and stopped publishing
- * @property {Object} lastReadingAt - last_reading_at: raw, ago, and parsed
- * @property {Object} createdAt - created_at: raw, ago, and parsed
- * @property {Object} updatedAt - updated_at: raw, ago, and parsed
- * @property {Object} location - Location of device. Object with lat, long, elevation, city, country, country_code
- * @property {string} locationString - Location of device. Ex: Madrid, Spain; Germany; Paris, France
- * @property {Object} hardware - Device hardware field. Contains type, version, info, slug and name
- * @property {string} hardwareName - Device hardware name
- * @property {bool} isLegacy - True if legacy device
- * @property {bool} isSCK - True if SC device
- * @property {string} avatar - URL that contains the user avatar
- */
- function Device(object) {
- // Basic information
- this.id = object.id;
- this.name = object.name;
- this.state = deviceUtils.parseState(object);
- this.description = object.description;
- this.token = object.device_token;
- this.macAddress = object.mac_address;
+ // Taken from this answer on SO:
+ // https://stackoverflow.com/questions/17893708/angularjs-textarea-bind-to-json-object-shows-object-object
+ angular.module('app.components').directive('jsonText', function() {
+ return {
+ restrict: 'A',
+ require: 'ngModel',
+ link: function(scope, element, attr, ngModel){
+ function into(input) {
+ return JSON.parse(input);
+ }
+ function out(data) {
+ return JSON.stringify(data);
+ }
+ ngModel.$parsers.push(into);
+ ngModel.$formatters.push(out);
+ }
+ };
+ });
- // Tags and dates
- this.systemTags = deviceUtils.parseSystemTags(object);
- this.userTags = deviceUtils.parseUserTags(object);
- this.isPrivate = deviceUtils.isPrivate(object);
- this.preciseLocation = deviceUtils.preciseLocation(object);
- this.enableForwarding = deviceUtils.enableForwarding(object);
- this.notifications = deviceUtils.parseNotifications(object);
- this.lastReadingAt = timeUtils.parseDate(object.last_reading_at);
- this.createdAt = timeUtils.parseDate(object.created_at);
- this.updatedAt = timeUtils.parseDate(object.updated_at);
+ angular.module('app.components')
+ .controller('EditKitController', EditKitController);
- // Location
- this.location = object.location;
- this.locationString = deviceUtils.parseLocation(object);
+ EditKitController.$inject = ['$scope', '$element', '$location', '$timeout', '$state',
+ 'animation','auth','device', 'tag', 'alert', 'step', '$stateParams', 'FullDevice'];
+ function EditKitController($scope, $element, $location, $timeout, $state, animation,
+ auth, device, tag, alert, step, $stateParams, FullDevice) {
- // Hardware
- this.hardware = deviceUtils.parseHardware(object);
- this.hardwareName = deviceUtils.parseHardwareName(this);
- this.isLegacy = deviceUtils.isLegacyVersion(this);
- this.isSCK = deviceUtils.isSCKHardware(this);
- // this.class = deviceUtils.classify(object); // TODO - Do we need this?
+ var vm = this;
- this.avatar = deviceUtils.parseAvatar();
- /*jshint camelcase: false */
- }
+ // WAIT INTERVAL FOR USER FEEDBACK and TRANSITIONS (This will need to change)
+ var timewait = {
+ long: 5000,
+ normal: 2000,
+ short: 1000
+ };
- return Device;
- }]);
-})();
+ vm.step = step;
-(function() {
- 'use strict';
+ // KEY USER ACTIONS
+ vm.submitFormAndKit = submitFormAndKit;
+ vm.backToProfile = backToProfile;
+ vm.backToDevice = backToDevice;
+ vm.submitForm = submitForm;
+ vm.goToStep = goToStep;
+ vm.nextAction = 'save';
- angular.module('app.components')
- .directive('noDataBackdrop', noDataBackdrop);
+ // EXPOSURE SELECT
+ vm.exposure = [
+ {name: 'indoor', value: 1},
+ {name: 'outdoor', value: 2}
+ ];
- /**
- * Backdrop for chart section when kit has no data
- *
- */
- noDataBackdrop.$inject = [];
+ // FORM INFO
+ vm.deviceForm = {};
+ vm.device = undefined;
- function noDataBackdrop() {
- return {
- restrict: 'A',
- scope: {},
- templateUrl: 'app/core/animation/backdrop/noDataBackdrop.html',
- controller: function($scope, $timeout) {
- var vm = this;
+ $scope.clearSearchTerm = function() {
+ $scope.searchTerm = '';
+ };
+ // The md-select directive eats keydown events for some quick select
+ // logic. Since we have a search input here, we don't need that logic.
+ $element.find('input').on('keydown', function(ev) {
+ ev.stopPropagation();
+ });
- vm.deviceWithoutData = false;
- vm.scrollToComments = scrollToComments;
+ $scope.$on('leafletDirectiveMarker.dragend', function(event, args){
+ vm.deviceForm.location.lat = args.model.lat;
+ vm.deviceForm.location.lng = args.model.lng;
+ });
- $scope.$on('deviceWithoutData', function(ev, data) {
+ // MAP CONFIGURATION
+ var mapBoxToken = 'pk.eyJ1IjoidG9tYXNkaWV6IiwiYSI6ImRTd01HSGsifQ.loQdtLNQ8GJkJl2LUzzxVg';
- $timeout(function() {
- vm.device = data.device;
- vm.deviceWithoutData = true;
-
- if (data.belongsToUser) {
- vm.user = 'owner';
- } else {
- vm.user = 'visitor';
- }
- }, 0);
-
- });
-
- function scrollToComments(){
- location.hash = '';
- location.hash = '#disqus_thread';
- }
- },
- controllerAs: 'vm'
- };
- }
-})();
+ vm.getLocation = getLocation;
+ vm.markers = {};
+ vm.tiles = {
+ url: 'https://api.mapbox.com/styles/v1/mapbox/streets-v10/tiles/{z}/{x}/{y}?access_token=' + mapBoxToken
+ };
+ vm.defaults = {
+ scrollWheelZoom: false
+ };
-(function() {
- 'use strict';
+ initialize();
- angular.module('app.components')
- .directive('loadingBackdrop', loadingBackdrop);
+ /////////////////
- /**
- * Backdrop for app initialization and between states
- *
- */
- loadingBackdrop.$inject = [];
- function loadingBackdrop() {
- return {
- templateUrl: 'app/core/animation/backdrop/loadingBackdrop.html',
- controller: function($scope) {
- var vm = this;
- vm.isViewLoading = true;
- vm.mapStateLoading = false;
+ function initialize() {
+ var deviceID = $stateParams.id;
- // listen for app loading event
- $scope.$on('viewLoading', function() {
- vm.isViewLoading = true;
- });
+ animation.viewLoaded();
+ getTags();
- $scope.$on('viewLoaded', function() {
- vm.isViewLoading = false;
- });
+ if (!deviceID || deviceID === ''){
+ return;
+ }
+ device.getDevice(deviceID)
+ .then(function(deviceData) {
+ vm.device = new FullDevice(deviceData);
+ vm.userRole = auth.getCurrentUser().data.role;
+ vm.deviceForm = {
+ name: vm.device.name,
+ exposure: findExposureFromLabels(vm.device.systemTags),
+ location: {
+ lat: vm.device.location.latitude,
+ lng: vm.device.location.longitude,
+ zoom: 16
+ },
+ is_private: vm.device.isPrivate,
+ precise_location: vm.device.preciseLocation,
+ enable_forwarding: vm.device.enableForwarding,
+ notify_low_battery: vm.device.notifications.lowBattery,
+ notify_stopped_publishing: vm.device.notifications.stopPublishing,
+ tags: vm.device.userTags,
+ postprocessing: vm.device.postProcessing,
+ description: vm.device.description,
+ hardwareName: vm.device.hardware.name
+ };
+ vm.markers = {
+ main: {
+ lat: vm.device.location.latitude,
+ lng: vm.device.location.longitude,
+ draggable: true
+ }
+ };
- // listen for map state loading event
- $scope.$on('mapStateLoading', function() {
- if(vm.isViewLoading) {
- return;
+ if (vm.device.isLegacy) {
+ vm.deviceForm.macAddress = vm.device.macAddress;
}
- vm.mapStateLoading = true;
- });
-
- $scope.$on('mapStateLoaded', function() {
- vm.mapStateLoading = false;
});
- },
- controllerAs: 'vm'
- };
- }
-})();
+ }
-(function() {
- 'use strict';
+ // Return tags in a comma separated list
+ function joinSelectedTags(){
+ let tmp = []
+ $scope.selectedTags.forEach(function(e){
+ tmp.push(e.name)
+ })
+ return tmp.join(', ');
+ }
- angular.module('app.components')
- .controller('UserProfileController', UserProfileController);
+ function getLocation() {
+ window.navigator.geolocation.getCurrentPosition(function(position) {
+ $scope.$apply(function() {
+ var lat = position.coords.latitude;
+ var lng = position.coords.longitude;
+ vm.deviceForm.location.lat = lat;
+ vm.deviceForm.location.lng = lng;
+ vm.markers.main.lat = lat;
+ vm.markers.main.lng = lng;
+ });
+ });
+ }
- UserProfileController.$inject = ['$scope', '$stateParams', '$location',
- 'user', 'auth', 'userUtils', '$timeout', 'animation',
- 'NonAuthUser', '$q', 'PreviewDevice'];
- function UserProfileController($scope, $stateParams, $location,
- user, auth, userUtils, $timeout, animation,
- NonAuthUser, $q, PreviewDevice) {
+ function submitFormAndKit(){
+ submitForm(backToProfile, timewait.normal);
+ }
- var vm = this;
- var userID = parseInt($stateParams.id);
+ function submitForm(next, delayTransition) {
+ var data = {
+ name: vm.deviceForm.name,
+ description: vm.deviceForm.description,
+ postprocessing_attributes: vm.deviceForm.postprocessing,
+ exposure: findExposure(vm.deviceForm.exposure),
+ latitude: vm.deviceForm.location.lat,
+ longitude: vm.deviceForm.location.lng,
+ is_private: vm.deviceForm.is_private,
+ enable_forwarding: vm.deviceForm.enable_forwarding,
+ precise_location: vm.deviceForm.precise_location,
+ notify_low_battery: vm.deviceForm.notify_low_battery,
+ notify_stopped_publishing: vm.deviceForm.notify_stopped_publishing,
+ mac_address: "",
+ /*jshint camelcase: false */
+ user_tags: joinSelectedTags(),
+ };
- vm.status = undefined;
- vm.user = {};
- vm.devices = [];
- vm.filteredDevices = [];
- vm.filterDevices = filterDevices;
+ vm.errors={};
- $scope.$on('loggedIn', function() {
- var authUser = auth.getCurrentUser().data;
- if( userUtils.isAuthUser(userID, authUser) ) {
- $location.path('/profile');
+ if(!vm.device.isSCK) {
+ data.hardware_name_override = vm.deviceForm.hardwareName;
}
- });
-
- initialize();
-
- //////////////////
-
- function initialize() {
- user.getUser(userID)
- .then(function(user) {
- vm.user = new NonAuthUser(user);
-
- if(!vm.user.devices.length) {
- return [];
- }
+ // Workaround for the mac_address bypass
+ // If mac_address is "", we get an error on the request -> we use it for the newKit
+ // If mac_address is null, no problem -> we use it for the
+ if ($stateParams.step === "2") {
+ data.mac_address = vm.deviceForm.macAddress ? vm.deviceForm.macAddress : "";
+ } else {
+ data.mac_address = vm.deviceForm.macAddress ? vm.deviceForm.macAddress : null;
+ }
- $q.all(vm.devices = vm.user.devices.map(function(data){
- return new PreviewDevice(data);
- }))
+ device.updateDevice(vm.device.id, data)
+ .then(
+ function() {
- }).then(function(error) {
- if(error && error.status === 404) {
- $location.url('/404');
- }
- });
+ if (next){
+ alert.success('Your kit was updated!');
+ }
- $timeout(function() {
- setSidebarMinHeight();
- animation.viewLoaded();
- }, 500);
+ device.updateContext().then(function(){
+ if (next){
+ $timeout(next, delayTransition);
+ }
+ });
+ })
+ .catch(function(err) {
+ if(err.data.errors) {
+ vm.errors = err.data.errors;
+ var message = Object.keys(vm.errors).map(function (key, _) {
+ return [key, vm.errors[key][0]].join(' '); }).join('');
+ alert.error('Oups! Check the input. Something went wrong!');
+ throw new Error('[Client:error] ' + message);
+ }
+ $timeout(function(){ }, timewait.long);
+ });
}
- function filterDevices(status) {
- if(status === 'all') {
- status = undefined;
+ function findExposureFromLabels(labels){
+ var label = vm.exposure.filter(function(n) {
+ return labels.indexOf(n.name) !== -1;
+ })[0];
+ if(label) {
+ return findExposure(label.name);
+ } else {
+ return findExposure(vm.exposure[0].name);
}
- vm.status = status;
}
- function setSidebarMinHeight() {
- var height = document.body.clientHeight / 4 * 3;
- angular.element('.profile_content').css('min-height', height + 'px');
- }
- }
-})();
+ function findExposure(nameOrValue) {
+ var findProp, resultProp;
-(function() {
- 'use strict';
+ //if it's a string
+ if(isNaN(parseInt(nameOrValue))) {
+ findProp = 'name';
+ resultProp = 'value';
+ } else {
+ findProp = 'value';
+ resultProp = 'name';
+ }
- angular.module('app.components')
- .controller('UploadController', UploadController);
+ var option = _.find(vm.exposure, function(exposureFromList) {
+ return exposureFromList[findProp] === nameOrValue;
+ });
+ if(option) {
+ return option[resultProp];
+ } else {
+ return vm.exposure[0][resultProp];
+ }
+ }
- UploadController.$inject = ['kit', '$state', '$stateParams', 'animation'];
- function UploadController(kit, $state, $stateParams, animation) {
- var vm = this;
+ function getTags() {
+ tag.getTags()
+ .then(function(tagsData) {
+ vm.tags = tagsData;
+ });
+ }
- vm.kit = kit;
+ function backToProfile(){
+ $state.transitionTo('layout.myProfile.kits', $stateParams,
+ { reload: false,
+ inherit: false,
+ notify: true
+ });
+ }
- vm.backToProfile = backToProfile;
+ function backToDevice(){
+ $state.transitionTo('layout.home.kit', $stateParams,
+ { reload: false,
+ inherit: false,
+ notify: true
+ });
+ }
- initialize();
+ function goToStep(step) {
+ vm.step = step;
+ $state.transitionTo('layout.kitEdit', { id:$stateParams.id, step: step} ,
+ {
+ reload: false,
+ inherit: false,
+ notify: false
+ });
+ }
+ }
+})();
- /////////////////
+(function() {
+ 'use strict';
- function initialize() {
- animation.viewLoaded();
- }
+ angular.module('app.components')
+ .factory('userUtils', userUtils);
- function backToProfile() {
- $state.transitionTo('layout.myProfile.kits', $stateParams,
- { reload: false,
- inherit: false,
- notify: true
- });
+ function userUtils() {
+ var service = {
+ isAdmin: isAdmin,
+ isAuthUser: isAuthUser
+ };
+ return service;
+
+ ///////////
+
+ function isAdmin(userData) {
+ return userData.role === 'admin';
+ }
+ function isAuthUser(userID, authUserData) {
+ return userID === authUserData.id;
+ }
}
- }
})();
-(function(){
-'use strict';
+(function() {
+ 'use strict';
+ angular.module('app.components')
+ .factory('timeUtils', timeUtils);
+ function timeUtils() {
+ var service = {
+ getSecondsFromDate: getSecondsFromDate,
+ getMillisFromDate: getMillisFromDate,
+ getCurrentRange: getCurrentRange,
+ getToday: getToday,
+ getHourBefore: getHourBefore,
+ getSevenDaysAgo: getSevenDaysAgo,
+ getDateIn: getDateIn,
+ convertTime: convertTime,
+ formatDate: formatDate,
+ isSameDay: isSameDay,
+ isWithin15min: isWithin15min,
+ isWithin1Month: isWithin1Month,
+ isWithin: isWithin,
+ isDiffMoreThan15min: isDiffMoreThan15min,
+ parseDate: parseDate
+ };
+ return service;
-function parseDataForPost(csvArray) {
- /*
- EXPECTED PAYLOAD
- {
- "data": [{
- "recorded_at": "2016-06-08 10:30:00",
- "sensors": [{
- "id": 22,
- "value": 21
- }]
- }]
- }
- */
- const ids = csvArray[3]; // save ids from the 4th header
- csvArray.splice(0,4); // remove useless headers
- return {
- data: csvArray.map((data) => {
- return {
- recorded_at: data.shift(), // get the timestamp from the first column
- sensors: data.map((value, index) => {
- return {
- id: ids[index+1], // get ID of sensor from headers
- value: value
- };
- })
- .filter((sensor) => sensor.value && sensor.id) // remove empty value or id
- };
- })
- };
-}
+ ////////////
+ function getDateIn(timeMS, format) {
+ if(!format) {
+ return timeMS;
+ }
+ var result;
+ if(format === 'ms') {
+ result = timeMS;
+ } else if(format === 's') {
+ result = timeMS / 1000;
+ } else if(format === 'm') {
+ result = timeMS / 1000 / 60;
+ } else if(format === 'h') {
+ result = timeMS / 1000 / 60 / 60;
+ } else if(format === 'd') {
+ result = timeMS / 1000 / 60 / 60 / 24;
+ }
+ return result;
+ }
-controller.$inject = ['device', 'Papa', '$mdDialog', '$q'];
-function controller(device, Papa, $mdDialog, $q) {
- var vm = this;
- vm.loadingStatus = false;
- vm.loadingProgress = 0;
- vm.loadingType = 'indeterminate';
- vm.csvFiles = [];
- vm.$onInit = function() {
- vm.kitLastUpdate = Math.floor(new Date(vm.kit.time).getTime() / 1000);
- }
- vm.onSelect = function() {
- vm.loadingStatus = true;
- vm.loadingType = 'indeterminate';
- }
- vm.change = function(files, invalidFiles) {
- let count = 0;
- vm.invalidFiles = invalidFiles;
- if (!files) { return; }
- vm.loadingStatus = true;
- vm.loadingType = 'determinate';
- vm.loadingProgress = 0;
- $q.all(
- files
- .filter((file) => vm._checkDuplicate(file))
- .map((file, index, filteredFiles) => {
- vm.csvFiles.push(file);
- return vm._analyzeData(file)
- .then((result) => {
- if (result.errors && result.errors.length > 0) {
- file.parseErrors = result.errors;
- }
- const lastTimestamp = Math.floor((new Date(result.data[result.data.length - 1][0])).getTime() / 1000);
- const isNew = vm.kitLastUpdate < lastTimestamp;
- file.checked = isNew;
- file.progress = null;
- file.isNew = isNew;
- })
- .then(() => {
- count += 1;
- vm.loadingProgress = (count)/filteredFiles.length * 100;
+ function convertTime(time) {
+ return moment(time).toISOString();
+ }
- });
- })
- ).then(() => {
- vm.loadingStatus = false;
- }).catch(() => {
- vm.loadingStatus = false;
- });
- }
+ function formatDate(time) {
+ return moment(time).format('YYYY-MM-DDTHH:mm:ss');
+ }
- vm.haveSelectedFiles = function() {
- return vm.csvFiles && vm.csvFiles.some((file) => file.checked);
- };
+ function getSecondsFromDate(date) {
+ return (new Date(date)).getTime();
+ }
- vm.haveSelectedNoFiles = function() {
- return vm.csvFiles && !vm.csvFiles.some((file) => file.checked);
- };
+ function getMillisFromDate(date) {
+ return (new Date(date)).getTime();
+ }
- vm.haveSelectedAllFiles = function() {
- return vm.csvFiles && vm.csvFiles.every((file) => file.checked);
- };
+ function getCurrentRange(fromDate, toDate) {
+ return moment(toDate).diff(moment(fromDate), 'days');
+ }
- vm.doAction = function() {
- switch (vm.action) {
- case 'selectAll':
- vm.selectAll(true);
- break;
- case 'deselectAll':
- vm.selectAll(false);
- break;
- case 'upload':
- vm.uploadData();
- break;
- case 'remove':
- vm.csvFiles = vm.csvFiles.filter((file) => !file.checked);
- break;
+ function getToday() {
+ return (new Date()).getTime();
}
- vm.action = null;
- };
- vm.selectAll = function(value) {
- vm.csvFiles.forEach((file) => { file.checked = value });
- };
+ function getSevenDaysAgo() {
+ return getSecondsFromDate( getToday() - (7 * 24 * 60 * 60 * 1000) );
+ }
- vm.removeFile = function(index) {
- vm.csvFiles.splice(index, 1);
- };
- vm._analyzeData = function(file) {
- file.progress = true;
- return Papa.parse(file, {
- delimiter: ',',
- dynamicTyping: true,
- worker: false,
- skipEmptyLines: true
- }).catch((err) => {
- file.progress = null;
- console('catch',err)
- });
- };
+ function getHourBefore(date) {
+ var now = moment(date);
+ return now.subtract(1, 'hour').valueOf();
+ }
- vm._checkDuplicate = function(file) {
- if (vm.csvFiles.some((csvFile) => file.name === csvFile.name)) {
- file.$errorMessages = {};
- file.$errorMessages.duplicate = true;
- vm.invalidFiles.push(file);
+ function isSameDay(day1, day2) {
+ day1 = moment(day1);
+ day2 = moment(day2);
+
+ if(day1.startOf('day').isSame(day2.startOf('day'))) {
+ return true;
+ }
return false;
- } else {
- return true;
}
- };
- vm.showErrorModal = function(csvFile) {
- $mdDialog.show({
- hasBackdrop: true,
- controller: ['$mdDialog',function($mdDialog) {
- this.parseErrors = csvFile.parseErrors;
- this.backEndErrors = csvFile.backEndErrors;
- this.cancel = function() { $mdDialog.hide(); };
- }],
- controllerAs: 'csvFile',
- templateUrl: 'app/components/upload/errorModal.html',
- clickOutsideToClose: true
- });
- }
+ function isDiffMoreThan15min(dateToCheckFrom, dateToCheckTo) {
+ var duration = moment.duration(moment(dateToCheckTo).diff(moment(dateToCheckFrom)));
+ return duration.as('minutes') > 15;
+ }
+ function isWithin15min(dateToCheck) {
+ var fifteenMinAgo = moment().subtract(15, 'minutes').valueOf();
+ dateToCheck = moment(dateToCheck).valueOf();
- vm.uploadData = function() {
- vm.loadingStatus = true;
- vm.loadingType = 'indeterminate';
- vm.loadingProgress = 0;
- let count = 0;
+ return dateToCheck > fifteenMinAgo;
+ }
- $q.all(
- vm.csvFiles
- .filter((file) => file.checked && !file.success)
- .map((file, index, filteredFiles) => {
- file.progress = true;
- return vm._analyzeData(file)
- .then((result) => parseDataForPost(result.data)) // TODO: Improvement remove
- // TODO: Improvement with workers
- .then((payload) => device.postReadings(vm.kit, payload))
- .then(() => {
- if (vm.loadingType === 'indeterminate') { vm.loadingType = 'determinate'; };
- file.success = true;
- file.progress = null;
- count += 1;
- vm.loadingProgress = (count)/filteredFiles.length * 100;
- })
- .catch((errors) => {
- console.log(errors);
- file.detailShowed = true;
- file.backEndErrors = errors;
- file.progress = null;
- });
- })
- ).then(() => {
- vm.loadingStatus = false;
- })
- .catch(() => {
- vm.loadingStatus = false;
- });
- }
-};
+ function isWithin1Month(dateToCheck) {
+ var oneMonthAgo = moment().subtract(1, 'months').valueOf();
+ dateToCheck = moment(dateToCheck).valueOf();
+ return dateToCheck > oneMonthAgo;
+ }
-angular.module('app.components')
- .component('scCsvUpload', {
- templateUrl: 'app/components/upload/csvUpload.html',
- controller: controller,
- bindings: {
- kit: '<'
- },
- controllerAs: 'vm'
- });
+ function isWithin(number, type, dateToCheck) {
+ var ago = moment().subtract(number, type).valueOf();
+ dateToCheck = moment(dateToCheck).valueOf();
+
+ return dateToCheck > ago;
+ }
+
+ function parseDate(object){
+ var time = object;
+ return {
+ raw: time,
+ parsed: !time ? 'No time' : moment(time).format('MMMM DD, YYYY - HH:mm'),
+ ago: !time ? 'No time' : moment(time).fromNow()
+ }
+ }
+ }
})();
(function() {
'use strict';
angular.module('app.components')
- .controller('tagsController', tagsController);
-
- tagsController.$inject = ['tag', '$scope', 'device', '$state', '$q',
- 'PreviewDevice', 'animation'
- ];
+ .factory('sensorUtils', sensorUtils);
- function tagsController(tag, $scope, device, $state, $q, PreviewDevice,
- animation) {
+ sensorUtils.$inject = ['timeUtils'];
+ function sensorUtils(timeUtils) {
+ var service = {
+ getRollup: getRollup,
+ getSensorName: getSensorName,
+ getSensorValue: getSensorValue,
+ getSensorPrevValue: getSensorPrevValue,
+ getSensorIcon: getSensorIcon,
+ getSensorArrow: getSensorArrow,
+ getSensorColor: getSensorColor,
+ getSensorDescription: getSensorDescription
+ };
+ return service;
- var vm = this;
+ ///////////////
- vm.selectedTags = tag.getSelectedTags();
- vm.markers = [];
- vm.kits = [];
- vm.percActive = 0;
+ function getRollup(dateFrom, dateTo) {
- initialize();
+ // Calculate how many data points we can fit on a users screen
+ // Smaller screens request less data from the API
+ var durationInSec = moment(dateTo).diff(moment(dateFrom)) / 1000;
+ var chartWidth = window.innerWidth / 2;
- /////////////////////////////////////////////////////////
+ var rollup = parseInt(durationInSec / chartWidth) + 's';
- function initialize() {
- if(vm.selectedTags.length === 0){
- $state.transitionTo('layout.home.kit');
+ /*
+ //var rangeDays = timeUtils.getCurrentRange(dateFrom, dateTo, {format: 'd'});
+ var rollup;
+ if(rangeDays <= 1) {
+ rollup = '15s';
+ } else if(rangeDays <= 7) {
+ rollup = '1h';//rollup = '15m';
+ } else if(rangeDays > 7) {
+ rollup = '1d';
+ }
+ */
+ return rollup;
}
- if (device.getWorldMarkers()) {
- // If the user has already loaded a prev page and has markers in mem or localstorage
- updateSelectedTags();
- } else {
- // If the user is new we wait the map to load the markers
- $scope.$on('mapStateLoaded', function(event, data) {
- updateSelectedTags();
- });
+ function getSensorName(name) {
+
+ var sensorName;
+ // TODO: Improvement check how we set new names
+ if( new RegExp('custom circuit', 'i').test(name) ) {
+ sensorName = name;
+ } else {
+ if(new RegExp('noise', 'i').test(name) ) {
+ sensorName = 'SOUND';
+ } else if(new RegExp('light', 'i').test(name) ) {
+ sensorName = 'LIGHT';
+ } else if((new RegExp('nets', 'i').test(name) ) ||
+ (new RegExp('wifi', 'i').test(name))) {
+ sensorName = 'NETWORKS';
+ } else if(new RegExp('co', 'i').test(name) ) {
+ sensorName = 'CO';
+ } else if(new RegExp('no2', 'i').test(name) ) {
+ sensorName = 'NO2';
+ } else if(new RegExp('humidity', 'i').test(name) ) {
+ sensorName = 'HUMIDITY';
+ } else if(new RegExp('temperature', 'i').test(name) ) {
+ sensorName = 'TEMPERATURE';
+ } else if(new RegExp('panel', 'i').test(name) ) {
+ sensorName = 'SOLAR PANEL';
+ } else if(new RegExp('battery', 'i').test(name) ) {
+ sensorName = 'BATTERY';
+ } else if(new RegExp('barometric pressure', 'i').test(name) ) {
+ sensorName = 'BAROMETRIC PRESSURE';
+ } else if(new RegExp('PM 1', 'i').test(name) ) {
+ sensorName = 'PM 1';
+ } else if(new RegExp('PM 2.5', 'i').test(name) ) {
+ sensorName = 'PM 2.5';
+ } else if(new RegExp('PM 10', 'i').test(name) ) {
+ sensorName = 'PM 10';
+ } else {
+ sensorName = name;
+ }
+ }
+ return sensorName.toUpperCase();
}
- }
+ function getSensorValue(sensor) {
+ var value = sensor.value;
- function updateSelectedTags(){
+ if(isNaN(parseInt(value))) {
+ value = 'NA';
+ } else {
+ value = round(value, 1).toString();
+ }
- vm.markers = tag.filterMarkersByTag(device.getWorldMarkers());
+ return value;
+ }
- var onlineMarkers = _.filter(vm.markers, isOnline);
- if (vm.markers.length === 0) {
- vm.percActive = 0;
- } else {
- vm.percActive = Math.floor(onlineMarkers.length / vm.markers.length *
- 100);
+ function round(value, precision) {
+ var multiplier = Math.pow(10, precision || 0);
+ return Math.round(value * multiplier) / multiplier;
}
- animation.viewLoaded();
+ function getSensorPrevValue(sensor) {
+ /*jshint camelcase: false */
+ var prevValue = sensor.prev_value;
+ return (prevValue && prevValue.toString() ) || 0;
+ }
- getTaggedDevices()
- .then(function(res){
- vm.kits = res;
- });
- }
+ function getSensorIcon(sensorName) {
+ var thisName = getSensorName(sensorName);
- function isOnline(marker) {
- return _.includes(marker.myData.labels, 'online');
- }
+ switch(thisName) {
+ case 'TEMPERATURE':
+ return './assets/images/temperature_icon_new.svg';
- function descLastUpdate(o) {
- return -new Date(o.last_reading_at).getTime();
- }
+ case 'HUMIDITY':
+ return './assets/images/humidity_icon_new.svg';
- function getTaggedDevices() {
+ case 'LIGHT':
+ return './assets/images/light_icon_new.svg';
- var deviceProm = _.map(vm.markers, getMarkerDevice);
+ case 'SOUND':
+ return './assets/images/sound_icon_new.svg';
- return $q.all(deviceProm)
- .then(function(devices) {
- return _.map(_.sortBy(devices, descLastUpdate), toPreviewDevice); // This sort is temp
- });
- }
+ case 'CO':
+ return './assets/images/co_icon_new.svg';
- function toPreviewDevice(dev) {
- return new PreviewDevice(dev);
- }
+ case 'NO2':
+ return './assets/images/no2_icon_new.svg';
- function getMarkerDevice(marker) {
- return device.getDevice(marker.myData.id);
- }
- }
+ case 'NETWORKS':
+ return './assets/images/networks_icon.svg';
-})();
+ case 'BATTERY':
+ return './assets/images/battery_icon.svg';
-(function(){
- 'use strict';
- angular.module('app.components')
- .directive('tag',tag);
+ case 'SOLAR PANEL':
+ return './assets/images/solar_panel_icon.svg';
- function tag(){
- return{
- restrict: 'E',
- scope:{
- tagName: '=',
- openTag: '&'
- },
- controller:function($scope, $state){
- $scope.openTag = function(){
- $state.go('layout.home.tags', {tags:[$scope.tagName]});
- };
- },
- template:'{{tagName}}',
- link: function(scope, element, attrs){
- element.addClass('tag');
+ case 'BAROMETRIC PRESSURE':
+ return './assets/images/pressure_icon_new.svg';
- if(typeof(attrs.clickable) !== 'undefined'){
- element.bind('click', scope.openTag);
+ case 'PM 1':
+ case 'PM 2.5':
+ case 'PM 10':
+ return './assets/images/particle_icon_new.svg';
+
+ default:
+ return './assets/images/unknownsensor_icon.svg';
}
}
- };
- }
-})();
-(function() {
- 'use strict';
+ function getSensorArrow(currentValue, prevValue) {
+ currentValue = parseInt(currentValue) || 0;
+ prevValue = parseInt(prevValue) || 0;
- angular.module('app.components')
- .controller('StoreModalController', StoreModalController);
+ if(currentValue > prevValue) {
+ return 'arrow_up';
+ } else if(currentValue < prevValue) {
+ return 'arrow_down';
+ } else {
+ return 'equal';
+ }
+ }
- StoreModalController.$inject = ['$scope', '$mdDialog'];
- function StoreModalController($scope, $mdDialog) {
+ function getSensorColor(sensorName) {
+ switch(getSensorName(sensorName)) {
+ case 'TEMPERATURE':
+ return '#FF3D4C';
- $scope.cancel = function() {
- $mdDialog.hide();
- };
- }
-})();
+ case 'HUMIDITY':
+ return '#55C4F5';
-(function() {
- 'use strict';
+ case 'LIGHT':
+ return '#ffc107';
- angular.module('app.components')
- .directive('store', store);
+ case 'SOUND':
+ return '#0019FF';
- function store() {
- return {
- scope: {
- isLoggedin: '=logged'
- },
- restrict: 'A',
- controller: 'StoreController',
- controllerAs: 'vm',
- templateUrl: 'app/components/store/store.html'
- };
- }
-})();
+ case 'CO':
+ return '#00A103';
-(function() {
- 'use strict';
+ case 'NO2':
+ return '#8cc252';
- angular.module('app.components')
- .controller('StoreController', StoreController);
+ case 'NETWORKS':
+ return '#681EBD';
- StoreController.$inject = ['$scope', '$mdDialog'];
- function StoreController($scope, $mdDialog) {
+ case 'SOLAR PANEL':
+ return '#d555ce';
- $scope.showStore = showStore;
+ case 'BATTERY':
+ return '#ff8601';
- $scope.$on('showStore', function() {
- showStore();
- });
-
- ////////////////
+ default:
+ return '#0019FF';
+ }
+ }
- function showStore() {
- $mdDialog.show({
- hasBackdrop: true,
- controller: 'StoreModalController',
- templateUrl: 'app/components/store/storeModal.html',
- clickOutsideToClose: true
- });
+ function getSensorDescription(sensorID, sensorTypes) {
+ return _(sensorTypes)
+ .chain()
+ .find(function(sensorType) {
+ return sensorType.id === sensorID;
+ })
+ .value()
+ .measurement.description;
+ }
}
-
- }
})();
(function() {
'use strict';
angular.module('app.components')
- .controller('StaticController', StaticController);
+ .factory('searchUtils', searchUtils);
- StaticController.$inject = ['$timeout', 'animation', '$mdDialog', '$location', '$anchorScroll'];
- function StaticController($timeout, animation, $mdDialog, $location, $anchorScroll) {
- var vm = this;
+ searchUtils.$inject = [];
+ function searchUtils() {
+ var service = {
+ parseLocation: parseLocation,
+ parseName: parseName,
+ parseIcon: parseIcon,
+ parseIconType: parseIconType
+ };
+ return service;
- vm.showStore = showStore;
+ /////////////////
- $anchorScroll.yOffset = 80;
+ function parseLocation(object) {
+ var location = '';
- ///////////////////////
+ if(!!object.city) {
+ location += object.city;
+ }
+ if(!!object.city && !!object.country) {
+ location += ', ';
+ }
+ if(!!object.country) {
+ location += object.country;
+ }
- initialize();
+ return location;
+ }
- //////////////////
+ function parseName(object) {
+ var name = object.type === 'User' ? object.username : object.name;
+ return name;
+ }
- function initialize() {
- $timeout(function() {
- animation.viewLoaded();
- if($location.hash()){
- $anchorScroll();
+ function parseIcon(object, type) {
+ switch(type) {
+ case 'User':
+ return object.profile_picture;
+ case 'Device':
+ return 'assets/images/kit.svg';
+ case 'Country':
+ case 'City':
+ return 'assets/images/location_icon_normal.svg';
}
- }, 500);
- }
+ }
- function showStore() {
- $mdDialog.show({
- hasBackdrop: true,
- controller: 'StoreModalController',
- templateUrl: 'app/components/store/storeModal.html',
- clickOutsideToClose: true
- });
+ function parseIconType(type) {
+ switch(type) {
+ case 'Device':
+ return 'div';
+ default:
+ return 'img';
+ }
+ }
}
- }
})();
(function() {
'use strict';
angular.module('app.components')
- .controller('SignupModalController', SignupModalController);
+ .factory('markerUtils', markerUtils);
- SignupModalController.$inject = ['$scope', '$mdDialog', 'user',
- 'alert', 'animation'];
- function SignupModalController($scope, $mdDialog, user,
- alert, animation ) {
- var vm = this;
- vm.answer = function(signupForm) {
+ markerUtils.$inject = ['deviceUtils', 'MARKER_ICONS'];
+ function markerUtils(deviceUtils, MARKER_ICONS) {
+ var service = {
+ getIcon: getIcon,
+ getMarkerIcon: getMarkerIcon,
+ };
+ _.defaults(service, deviceUtils);
+ return service;
- if (!signupForm.$valid){
- return;
+ ///////////////
+
+ function getIcon(object) {
+ var icon;
+ var labels = deviceUtils.parseSystemTags(object);
+ var isSCKHardware = deviceUtils.isSCKHardware(object);
+
+ if(hasLabel(labels, 'offline')) {
+ icon = MARKER_ICONS.markerSmartCitizenOffline;
+ } else if (isSCKHardware) {
+ icon = MARKER_ICONS.markerSmartCitizenOnline;
+ } else {
+ icon = MARKER_ICONS.markerExperimentalNormal;
}
+ return icon;
+ }
- $scope.waitingFromServer = true;
- user.createUser(vm.user)
- .then(function() {
- alert.success('Signup was successful');
- $mdDialog.hide();
- }).catch(function(err) {
- alert.error('Signup failed');
- $scope.errors = err.data.errors;
- })
- .finally(function() {
- $scope.waitingFromServer = false;
- });
- };
- $scope.hide = function() {
- $mdDialog.hide();
- };
- $scope.cancel = function() {
- $mdDialog.cancel();
- };
-
- $scope.openLogin = function() {
- animation.showLogin();
- $mdDialog.hide();
- };
- }
-})();
-
-(function() {
- 'use strict';
+ function hasLabel(labels, targetLabel) {
+ return _.some(labels, function(label) {
+ return label === targetLabel;
+ });
+ }
- angular.module('app.components')
- .directive('signup', signup);
+ function getMarkerIcon(marker, state) {
+ var markerType = marker.icon.className;
- function signup() {
- return {
- scope: {
- show: '=',
- },
- restrict: 'A',
- controller: 'SignupController',
- controllerAs: 'vm',
- templateUrl: 'app/components/signup/signup.html'
- };
+ if(state === 'active') {
+ marker.icon = MARKER_ICONS[markerType + 'Active'];
+ marker.focus = true;
+ } else if(state === 'inactive') {
+ var targetClass = markerType.split(' ')[0];
+ marker.icon = MARKER_ICONS[targetClass];
+ }
+ return marker;
+ }
}
})();
@@ -2440,208 +2315,307 @@ angular.module('app.components')
'use strict';
angular.module('app.components')
- .controller('SignupController', SignupController);
+ .factory('mapUtils', mapUtils);
- SignupController.$inject = ['$scope', '$mdDialog'];
- function SignupController($scope, $mdDialog) {
- var vm = this;
+ mapUtils.$inject = [];
+ function mapUtils() {
+ var service = {
+ getDefaultFilters: getDefaultFilters,
+ setDefaultFilters: setDefaultFilters,
+ canFilterBeRemoved: canFilterBeRemoved
+ };
+ return service;
- vm.showSignup = showSignup;
+ //////////////
- $scope.$on('showSignup', function() {
- showSignup();
- });
- ////////////////////////
+ function getDefaultFilters(filterData, defaultFilters) {
+ var obj = {};
+ if(!filterData.indoor && !filterData.outdoor) {
+ obj[defaultFilters.exposure] = true;
+ }
+ if(!filterData.online && !filterData.offline) {
+ obj[defaultFilters.status] = true;
+ }
+ return obj;
+ }
+ function setDefaultFilters(filterData) {
+ var obj = {};
+ if(!filterData.indoor || !filterData.outdoor) {
+ obj.exposure = filterData.indoor ? 'indoor' : 'outdoor';
+ }
+ if(!filterData.online || !filterData.offline) {
+ obj.status = filterData.online ? 'online' : 'offline';
+ }
+ return obj;
+ }
- function showSignup() {
- $mdDialog.show({
- fullscreen: true,
- hasBackdrop: true,
- controller: 'SignupModalController',
- controllerAs: 'vm',
- templateUrl: 'app/components/signup/signupModal.html',
- clickOutsideToClose: true
- });
+ function canFilterBeRemoved(filterData, filterName) {
+ if(filterName === 'indoor' || filterName === 'outdoor') {
+ return filterData.indoor && filterData.outdoor;
+ } else if(filterName === 'online' || filterName === 'offline') {
+ return filterData.online && filterData.offline;
+ }
}
}
})();
(function() {
-'use strict';
-
-
+ 'use strict';
angular.module('app.components')
- .directive('search', search);
+ .config(function ($provide) {
+ $provide.decorator('$exceptionHandler', ['$delegate', function($delegate) {
+ return function (exception, cause) {
+ /*jshint camelcase: false */
+ $delegate(exception, cause);
+ };
+ }]);
- function search() {
- return {
- scope: true,
- restrict: 'E',
- templateUrl: 'app/components/search/search.html',
- controller: 'SearchController',
- controllerAs: 'vm'
- };
- }
+ });
})();
(function() {
'use strict';
angular.module('app.components')
- .controller('SearchController', SearchController);
+ .factory('deviceUtils', deviceUtils);
- SearchController.$inject = ['$scope', 'search', 'SearchResult', '$location', 'animation', 'SearchResultLocation'];
- function SearchController($scope, search, SearchResult, $location, animation, SearchResultLocation) {
- var vm = this;
+ deviceUtils.$inject = ['COUNTRY_CODES', 'device'];
+ function deviceUtils(COUNTRY_CODES, device) {
+ var service = {
+ parseLocation: parseLocation,
+ parseCoordinates: parseCoordinates,
+ parseSystemTags: parseSystemTags,
+ parseUserTags: parseUserTags,
+ classify: classify,
+ parseNotifications: parseNotifications,
+ parseOwner: parseOwner,
+ parseName: parseName,
+ parseString: parseString,
+ parseHardware: parseHardware,
+ parseHardwareInfo: parseHardwareInfo,
+ parseHardwareName: parseHardwareName,
+ isPrivate: isPrivate,
+ preciseLocation: preciseLocation,
+ enableForwarding: enableForwarding,
+ isLegacyVersion: isLegacyVersion,
+ isSCKHardware: isSCKHardware,
+ parseState: parseState,
+ parseAvatar: parseAvatar,
+ belongsToUser: belongsToUser,
+ parseSensorTime: parseSensorTime
+ };
- vm.searchTextChange = searchTextChange;
- vm.selectedItemChange = selectedItemChange;
- vm.querySearch = querySearch;
+ return service;
- ///////////////////
+ ///////////////
- function searchTextChange() {
- }
+ function parseLocation(object) {
+ var location = '';
+ var city = '';
+ var country = '';
- function selectedItemChange(result) {
- if (!result) { return; }
- if(result.type === 'User') {
- $location.path('/users/' + result.id);
- } else if(result.type === 'Device') {
- $location.path('/kits/' + result.id);
- } else if (result.type === 'City'){
- animation.goToLocation({lat: result.lat, lng: result.lng, type: result.type, layer: result.layer});
+ if (object.location) {
+ city = object.location.city;
+ country = object.location.country;
+ if(!!city) {
+ location += city;
+ }
+ if(!!city && !!location) {
+ location += ', '
+ }
+ if(!!country) {
+ location += country;
+ }
}
+ return location;
}
- function querySearch(query) {
- if(query.length < 3) {
- return [];
+ function parseCoordinates(object) {
+ if (object.location) {
+ return {
+ lat: object.location.latitude,
+ lng: object.location.longitude
+ };
}
+ // TODO: Bug - what happens if no location?
+ }
- return search.globalSearch(query)
- .then(function(data) {
+ function parseSystemTags(object) {
+ /*jshint camelcase: false */
+ return object.system_tags;
+ }
- return data.map(function(object) {
+ function parseUserTags(object) {
+ return object.user_tags;
+ }
- if(object.type === 'City' || object.type === 'Country') {
- return new SearchResultLocation(object);
- } else {
- return new SearchResult(object);
- }
- });
- });
+ function parseNotifications(object){
+ return {
+ lowBattery: object.notify.low_battery,
+ stopPublishing: object.notify.stopped_publishing
+ }
}
- }
-})();
-(function() {
- 'use strict';
+ function classify(kitType) {
+ if(!kitType) {
+ return '';
+ }
+ return kitType.toLowerCase().split(' ').join('_');
+ }
- angular.module('app.components')
- .controller('PasswordResetController', PasswordResetController);
+ function parseName(object, trim=false) {
+ if(!object.name) {
+ return;
+ }
+ if (trim) {
+ return object.name.length <= 41 ? object.name : object.name.slice(0, 35).concat(' ... ');
+ }
+ return object.name;
+ }
- PasswordResetController.$inject = ['$mdDialog', '$stateParams', '$timeout',
- 'animation', '$location', 'alert', 'auth'];
- function PasswordResetController($mdDialog, $stateParams, $timeout,
- animation, $location, alert, auth) {
-
- var vm = this;
- vm.showForm = false;
- vm.form = {};
- vm.isDifferent = false;
- vm.answer = answer;
+ function parseHardware(object) {
+ if (!object.hardware) {
+ return;
+ }
- initialize();
- ///////////
+ return {
+ name: parseString(object.hardware.name),
+ type: parseString(object.hardware.type),
+ description: parseString(object.hardware.description),
+ version: parseVersionString(object.hardware.version),
+ slug: object.hardware.slug,
+ info: parseHardwareInfo(object.hardware.info)
+ }
+ }
- function initialize() {
- $timeout(function() {
- animation.viewLoaded();
- }, 500);
- getUserData();
+ function parseString(str) {
+ if (typeof(str) !== 'string') { return null; }
+ return str;
}
- function getUserData() {
- auth.getResetPassword($stateParams.code)
- .then(function() {
- vm.showForm = true;
- })
- .catch(function() {
- alert.error('Wrong url');
- $location.path('/');
- });
+ function parseVersionString (str) {
+ if (typeof(str) !== 'string') { return null; }
+ var x = str.split('.');
+ // parse from string or default to 0 if can't parse
+ var maj = parseInt(x[0]) || 0;
+ var min = parseInt(x[1]) || 0;
+ var pat = parseInt(x[2]) || 0;
+ return {
+ major: maj,
+ minor: min,
+ patch: pat
+ };
}
- function answer(data) {
- vm.waitingFromServer = true;
- vm.errors = undefined;
+ function parseHardwareInfo (object) {
+ if (!object) { return null; } // null
+ if (typeof(object) == 'string') { return null; } // FILTERED
- if(data.newPassword === data.confirmPassword) {
- vm.isDifferent = false;
+ var id = parseString(object.id);
+ var mac = parseString(object.mac);
+ var time = Date(object.time);
+ var esp_bd = parseString(object.esp_bd);
+ var hw_ver = parseString(object.hw_ver);
+ var sam_bd = parseString(object.sam_bd);
+ var esp_ver = parseString(object.esp_ver);
+ var sam_ver = parseString(object.sam_ver);
+
+ return {
+ id: id,
+ mac: mac,
+ time: time,
+ esp_bd: esp_bd,
+ hw_ver: hw_ver,
+ sam_bd: sam_bd,
+ esp_ver: esp_ver,
+ sam_ver: sam_ver
+ };
+ }
+
+ function parseHardwareName(object) {
+ if (object.hasOwnProperty('hardware')) {
+ if (!object.hardware.name) {
+ return 'Unknown hardware'
+ }
+ return object.hardware.name;
} else {
- vm.isDifferent = true;
- return;
+ return 'Unknown hardware'
}
+ }
- auth.patchResetPassword($stateParams.code, {password: data.newPassword})
- .then(function() {
- alert.success('Your data was updated successfully');
- $location.path('/profile');
- })
- .catch(function(err) {
- alert.error('Your data wasn\'t updated');
- vm.errors = err.data.errors;
- })
- .finally(function() {
- vm.waitingFromServer = false;
- });
+ function isPrivate(object) {
+ return object.data_policy.is_private;
}
- }
-})();
-(function() {
- 'use strict';
+ function preciseLocation(object) {
+ return object.data_policy.precise_location;
+ }
- angular.module('app.components')
- .controller('PasswordRecoveryModalController', PasswordRecoveryModalController);
+ function enableForwarding(object) {
+ return object.data_policy.enable_forwarding ;
+ }
- PasswordRecoveryModalController.$inject = ['$scope', 'animation', '$mdDialog', 'auth', 'alert'];
- function PasswordRecoveryModalController($scope, animation, $mdDialog, auth, alert) {
+ function isLegacyVersion (object) {
+ if (!object.hardware || !object.hardware.version || object.hardware.version.major > 1) {
+ return false;
+ } else {
+ if (object.hardware.version.major == 1 && object.hardware.version.minor <5 ){
+ return true;
+ }
+ return false;
+ }
+ }
- $scope.hide = function() {
- $mdDialog.hide();
- };
- $scope.cancel = function() {
- $mdDialog.cancel();
- };
+ function isSCKHardware (object){
+ if (!object.hardware || !object.hardware.type || object.hardware.type != 'SCK') {
+ return false;
+ } else {
+ return true;
+ }
+ }
- $scope.recoverPassword = function() {
- $scope.waitingFromServer = true;
- var data = {
+ function parseOwner(object) {
+ return {
+ id: object.owner.id,
+ username: object.owner.username,
/*jshint camelcase: false */
- email_or_username: $scope.input
+ devices: object.owner.device_ids,
+ city: object.owner.location.city,
+ country: COUNTRY_CODES[object.owner.location.country_code],
+ url: object.owner.url,
+ profile_picture: object.owner.profile_picture
};
+ }
- auth.recoverPassword(data)
- .then(function() {
- alert.success('You were sent an email to recover your password');
- $mdDialog.hide();
- })
- .catch(function(err) {
- alert.error('That username doesn\'t exist');
- $scope.errors = err.data;
- })
- .finally(function() {
- $scope.waitingFromServer = false;
- });
- };
+ function parseState(status) {
+ var name = parseStateName(status);
+ var className = classify(name);
- $scope.openSignup = function() {
- animation.showSignup();
- $mdDialog.hide();
- };
+ return {
+ name: name,
+ className: className
+ };
+ }
+
+ function parseStateName(object) {
+ return object.state.replace('_', ' ');
+ }
+
+ function parseAvatar() {
+ return './assets/images/sckit_avatar.jpg';
+ }
+
+ function parseSensorTime(sensor) {
+ /*jshint camelcase: false */
+ return moment(sensor.recorded_at).format('');
+ }
+
+ function belongsToUser(devicesArray, deviceID) {
+ return _.some(devicesArray, function(device) {
+ return device.id === deviceID;
+ });
+ }
}
})();
@@ -2649,1065 +2623,1111 @@ angular.module('app.components')
'use strict';
angular.module('app.components')
- .controller('PasswordRecoveryController', PasswordRecoveryController);
-
- PasswordRecoveryController.$inject = ['auth', 'alert', '$mdDialog'];
- function PasswordRecoveryController(auth, alert, $mdDialog) {
- var vm = this;
-
- vm.waitingFromServer = false;
- vm.errors = undefined;
- vm.recoverPassword = recoverPassword;
-
- ///////////////
+ .filter('filterLabel', filterLabel);
- function recoverPassword() {
- vm.waitingFromServer = true;
- vm.errors = undefined;
-
- var data = {
- username: vm.username
- };
- auth.recoverPassword(data)
- .then(function() {
- alert.success('You were sent an email to recover your password');
- $mdDialog.hide();
- })
- .catch(function(err) {
- vm.errors = err.data.errors;
- if(vm.errors) {
- alert.error('That email/username doesn\'t exist');
- }
- })
- .finally(function() {
- vm.waitingFromServer = false;
- });
- }
- }
+ function filterLabel() {
+ return function(devices, targetLabel) {
+ if(targetLabel === undefined) {
+ return devices;
+ }
+ if(devices) {
+ return _.filter(devices, function(device) {
+ var containsLabel = device.systemTags.indexOf(targetLabel) !== -1;
+ if(containsLabel) {
+ return containsLabel;
+ }
+ // This should be fixed or polished in the future
+ // var containsNewIfTargetIsOnline = targetLabel === 'online' && _.some(kit.labels, function(label) {return label.indexOf('new') !== -1;});
+ // return containsNewIfTargetIsOnline;
+ });
+ }
+ };
+ }
})();
(function() {
'use strict';
- angular.module('app.components')
- .controller('MyProfileController', MyProfileController);
+ /**
+ * Tools links for user profile
+ * @constant
+ * @type {Array}
+ */
- MyProfileController.$inject = ['$scope', '$location', '$q', '$interval',
- 'userData', 'AuthUser', 'user', 'auth', 'alert',
- 'COUNTRY_CODES', '$timeout', 'file', 'animation',
- '$mdDialog', 'PreviewDevice', 'device', 'deviceUtils',
- 'userUtils', '$filter', '$state', 'Restangular', '$window'];
- function MyProfileController($scope, $location, $q, $interval,
- userData, AuthUser, user, auth, alert,
- COUNTRY_CODES, $timeout, file, animation,
- $mdDialog, PreviewDevice, device, deviceUtils,
- userUtils, $filter, $state, Restangular, $window) {
-
- var vm = this;
-
- vm.unhighlightIcon = unhighlightIcon;
-
- //PROFILE TAB
- vm.formUser = {};
- vm.getCountries = getCountries;
-
- vm.user = userData;
- copyUserToForm(vm.formUser, vm.user);
- vm.searchText = vm.formUser.country;
-
- vm.updateUser = updateUser;
- vm.removeUser = removeUser;
- vm.uploadAvatar = uploadAvatar;
-
- //THIS IS TEMPORARY.
- // Will grow on to a dynamic API KEY management
- // with the new /accounts oAuth mgmt methods
-
- // The auth controller has not populated the `user` at this point,
- // so user.token is undefined
- // This controller depends on auth has already been run.
- vm.user.token = auth.getToken();
- vm.addNewDevice = addNewDevice;
-
- //KITS TAB
- vm.devices = [];
- vm.deviceStatus = undefined;
- vm.removeDevice = removeDevice;
- vm.downloadData = downloadData;
-
- vm.filteredDevices = [];
- vm.dropdownSelected = undefined;
-
- //SIDEBAR
- vm.filterDevices = filterDevices;
- vm.filterTools = filterTools;
-
- vm.selectThisTab = selectThisTab;
-
- $scope.$on('loggedOut', function() {
- $location.path('/');
- });
-
- $scope.$on('devicesContextUpdated', function(){
- var userData = auth.getCurrentUser().data;
- if(userData){
- vm.user = userData;
- }
- initialize();
- });
-
- initialize();
-
- //////////////////
-
- function initialize() {
-
- startingTab();
- if(!vm.user.devices.length) {
- vm.devices = [];
- animation.viewLoaded();
- } else {
+ angular.module('app.components')
+ .constant('PROFILE_TOOLS', [{
+ type: 'documentation',
+ title: 'How to connect your Smart Citizen Kit tutorial',
+ description: 'Adding a Smart Citizen Kit tutorial',
+ avatar: '',
+ href: 'http://docs.smartcitizen.me/#/start/adding-a-smart-citizen-kit'
+ }, {
+ type: 'documentation',
+ title: 'Download the latest Smart Citizen Kit Firmware',
+ description: 'The latest Arduino firmware for your kit',
+ avatar: '',
+ href: 'https://github.com/fablabbcn/Smart-Citizen-Kit/releases/latest'
+ }, {
+ type: 'documentation',
+ title: 'API Documentation',
+ description: 'Documentation for the new API',
+ avatar: '',
+ href: 'http://developer.smartcitizen.me/'
+ }, {
+ type: 'community',
+ title: 'Smart Citizen Forum',
+ description: 'Join the community discussion. Your feedback is important for us.',
+ avatar: '',
+ href:'http://forum.smartcitizen.me/'
+ }, {
+ type: 'documentation',
+ title: 'Smart Citizen Kit hardware details',
+ description: 'Visit the docs',
+ avatar: 'https://docs.smartcitizen.me/#/start/hardware'
+ }, {
+ type: 'documentation',
+ title: 'Style Guide',
+ description: 'Guidelines of the Smart Citizen UI',
+ avatar: '',
+ href: '/styleguide'
+ }, {
+ type: 'social',
+ title: 'Like us on Facebook',
+ description: 'Join the community on Facebook',
+ avatar: '',
+ href: 'https://www.facebook.com/smartcitizenBCN'
+ }, {
+ type: 'social',
+ title: 'Follow us on Twitter',
+ description: 'Follow our news on Twitter',
+ avatar: '',
+ href: 'https://twitter.com/SmartCitizenKit'
+ }]);
+})();
- vm.devices = vm.user.devices.map(function(data) {
- return new PreviewDevice(data);
- })
+(function() {
+ 'use strict';
- $timeout(function() {
- mapWithBelongstoUser(vm.devices);
- filterDevices(vm.status);
- setSidebarMinHeight();
- animation.viewLoaded();
- });
+ /**
+ * Marker icons
+ * @constant
+ * @type {Object}
+ */
- }
+ angular.module('app.components')
+ .constant('MARKER_ICONS', {
+ defaultIcon: {},
+ markerSmartCitizenNormal: {
+ type: 'div',
+ className: 'markerSmartCitizenNormal',
+ iconSize: [24, 24]
+ },
+ markerExperimentalNormal: {
+ type: 'div',
+ className: 'markerExperimentalNormal',
+ iconSize: [24, 24]
+ },
+ markerSmartCitizenOnline: {
+ type: 'div',
+ className: 'markerSmartCitizenOnline',
+ iconSize: [24, 24]
+ },
+ markerSmartCitizenOnlineActive: {
+ type: 'div',
+ className: 'markerSmartCitizenOnline marker_blink',
+ iconSize: [24, 24]
+ },
+ markerSmartCitizenOffline: {
+ type: 'div',
+ className: 'markerSmartCitizenOffline',
+ iconSize: [24, 24]
+ },
+ markerSmartCitizenOfflineActive: {
+ type: 'div',
+ className: 'markerSmartCitizenOffline marker_blink',
+ iconSize: [24, 24]
}
+ });
+})();
- function filterDevices(status) {
- if(status === 'all') {
- status = undefined;
- }
- vm.deviceStatus = status;
- vm.filteredDevices = $filter('filterLabel')(vm.devices, vm.deviceStatus);
- }
+(function() {
+ 'use strict';
- function filterTools(type) {
- if(type === 'all') {
- type = undefined;
- }
- vm.toolType = type;
- }
+ /**
+ * Dropdown options for user
+ * @constant
+ * @type {Array}
+ */
+ angular.module('app.components')
+ .constant('DROPDOWN_OPTIONS_USER', [
+ {divider: true, text: 'Hi,', href: './profile'},
+ {text: 'My profile', href: './profile'},
+ {text: 'Log out', href: './logout'}
+ ]);
+})();
- function updateUser(userData) {
- if(userData.country) {
- _.each(COUNTRY_CODES, function(value, key) {
- if(value === userData.country) {
- /*jshint camelcase: false */
- userData.country_code = key;
- return;
- }
- });
- } else {
- userData.country_code = null;
- }
+(function() {
+ 'use strict';
- user.updateUser(userData)
- .then(function(data) {
- var user = new AuthUser(data);
- _.extend(vm.user, user);
- auth.updateUser();
- vm.errors = {};
- alert.success('User updated');
- })
- .catch(function(err) {
- alert.error('User could not be updated ');
- vm.errors = err.data.errors;
- });
- }
+ /**
+ * Dropdown options for community button
+ * @constant
+ * @type {Array}
+ */
- function removeUser() {
- var confirm = $mdDialog.confirm()
- .title('Delete your account?')
- .textContent('Are you sure you want to delete your account?')
- .ariaLabel('')
- .ok('delete')
- .cancel('cancel')
- .theme('primary')
- .clickOutsideToClose(true);
-
- $mdDialog.show(confirm)
- .then(function(){
- return Restangular.all('').customDELETE('me')
- .then(function(){
- alert.success('Account removed successfully. Redirecting you…');
- $timeout(function(){
- auth.logout();
- $state.transitionTo('landing');
- }, 2000);
- })
- .catch(function(){
- alert.error('Error occurred trying to delete your account.');
- });
- });
- }
-
- function selectThisTab(iconIndex, uistate){
- /* This looks more like a hack but we need to workout how to properly use md-tab with ui-router */
-
- highlightIcon(iconIndex);
-
- if ($state.current.name.includes('myProfileAdmin')){
- var transitionState = 'layout.myProfileAdmin.' + uistate;
- $state.transitionTo(transitionState, {id: userData.id});
- } else {
- var transitionState = 'layout.myProfile.' + uistate;
- $state.transitionTo(transitionState);
- }
-
- }
-
- function startingTab() {
- /* This looks more like a hack but we need to workout how to properly use md-tab with ui-router */
-
- var childState = $state.current.name.split('.').pop();
-
- switch(childState) {
- case 'user':
- vm.startingTab = 1;
- break;
- default:
- vm.startingTab = 0;
- break;
- }
-
- }
-
- function highlightIcon(iconIndex) {
-
- var icons = angular.element('.myProfile_tab_icon');
-
- _.each(icons, function(icon) {
- unhighlightIcon(icon);
- });
-
- var icon = icons[iconIndex];
-
- angular.element(icon).find('.stroke_container').css({'stroke': 'white', 'stroke-width': '0.01px'});
- angular.element(icon).find('.fill_container').css('fill', 'white');
- }
-
- function unhighlightIcon(icon) {
- icon = angular.element(icon);
-
- icon.find('.stroke_container').css({'stroke': 'none'});
- icon.find('.fill_container').css('fill', '#FF8600');
- }
-
- function setSidebarMinHeight() {
- var height = document.body.clientHeight / 4 * 3;
- angular.element('.profile_content').css('min-height', height + 'px');
- }
-
- function getCountries(searchText) {
- return _.filter(COUNTRY_CODES, createFilter(searchText));
- }
-
- function createFilter(searchText) {
- searchText = searchText.toLowerCase();
- return function(country) {
- country = country.toLowerCase();
- return country.indexOf(searchText) !== -1;
- };
- }
-
- function uploadAvatar(fileData) {
- if(fileData && fileData.length) {
-
- // TODO: Improvement Is there a simpler way to patch the image to the API and use the response?
- // Something like:
- //Restangular.all('/me').patch(data);
- // Instead of doing it manually like here:
- var fd = new FormData();
- fd.append('profile_picture', fileData[0]);
- Restangular.one('/me')
- .withHttpConfig({transformRequest: angular.identity})
- .customPATCH(fd, '', undefined, {'Content-Type': undefined})
- .then(function(resp){
- vm.user.profile_picture = resp.profile_picture;
- })
- }
- }
-
- function copyUserToForm(formData, userData) {
- var props = {username: true, email: true, city: true, country: true, country_code: true, url: true, constructor: false};
-
- for(var key in userData) {
- if(props[key]) {
- formData[key] = userData[key];
- }
- }
- }
-
- function mapWithBelongstoUser(devices){
- _.map(devices, addBelongProperty);
- }
-
- function addBelongProperty(device){
- device.belongProperty = deviceBelongsToUser(device);
- return device;
- }
-
-
- function deviceBelongsToUser(device){
- if(!auth.isAuth() || !device || !device.id) {
- return false;
- }
- var deviceID = parseInt(device.id);
- var userData = ( auth.getCurrentUser().data ) ||
- ($window.localStorage.getItem('smartcitizen.data') &&
- new AuthUser( JSON.parse(
- $window.localStorage.getItem('smartcitizen.data') )));
-
- var belongsToUser = deviceUtils.belongsToUser(userData.devices, deviceID);
- var isAdmin = userUtils.isAdmin(userData);
-
- return isAdmin || belongsToUser;
- }
-
- function downloadData(device){
- $mdDialog.show({
- hasBackdrop: true,
- controller: 'DownloadModalController',
- controllerAs: 'vm',
- templateUrl: 'app/components/download/downloadModal.html',
- clickOutsideToClose: true,
- locals: {thisDevice:device}
- }).then(function(){
- var alert = $mdDialog.alert()
- .title('SUCCESS')
- .textContent('We are processing your data. Soon you will be notified in your inbox')
- .ariaLabel('')
- .ok('OK!')
- .theme('primary')
- .clickOutsideToClose(true);
-
- $mdDialog.show(alert);
- }).catch(function(err){
- if (!err){
- return;
- }
- var errorAlert = $mdDialog.alert()
- .title('ERROR')
- .textContent('Uh-oh, something went wrong')
- .ariaLabel('')
- .ok('D\'oh')
- .theme('primary')
- .clickOutsideToClose(false);
-
- $mdDialog.show(errorAlert);
- });
- }
-
- function removeDevice(deviceID) {
- var confirm = $mdDialog.confirm()
- .title('Delete this kit?')
- .textContent('Are you sure you want to delete this kit?')
- .ariaLabel('')
- .ok('DELETE')
- .cancel('Cancel')
- .theme('primary')
- .clickOutsideToClose(true);
-
- $mdDialog
- .show(confirm)
- .then(function(){
- device
- .removeDevice(deviceID)
- .then(function(){
- alert.success('Your kit was deleted successfully');
- device.updateContext();
- })
- .catch(function(){
- alert.error('Error trying to delete your kit.');
- });
- });
- }
-
- $scope.addDeviceSelector = addDeviceSelector;
- function addDeviceSelector(){
- $mdDialog.show({
- templateUrl: 'app/components/myProfile/addDeviceSelectorModal.html',
- clickOutsideToClose: true,
- multiple: true,
- controller: DialogController,
- });
- }
-
- function DialogController($scope, $mdDialog){
- $scope.cancel = function(){
- $mdDialog.cancel();
- };
- }
-
- function addNewDevice() {
- var confirm = $mdDialog.confirm()
- .title('Hey! Do you want to add a new kit?')
- .textContent('Please, notice this currently supports just the SCK 1.0 and SCK 1.1')
- .ariaLabel('')
- .ok('Ok')
- .cancel('Cancel')
- .theme('primary')
- .clickOutsideToClose(true);
-
- $mdDialog
- .show(confirm)
- .then(function(){
- $state.go('layout.kitAdd');
- });
- }
-
-
- }
-})();
+ angular.module('app.components')
+ .constant('DROPDOWN_OPTIONS_COMMUNITY', [
+ {text: 'About', href: '/about'},
+ {text: 'Forum', href: 'https://forum.smartcitizen.me/'},
+ {text: 'Documentation', href: 'http://docs.smartcitizen.me/'},
+ {text: 'API Reference', href: 'http://developer.smartcitizen.me/'},
+ {text: 'Github', href: 'https://github.com/fablabbcn/Smart-Citizen-Kit'},
+ {text: 'Legal', href: '/policy'}
+ ]);
+})();
(function() {
'use strict';
+ /**
+ * Country codes.
+ * @constant
+ * @type {Object}
+ */
+
angular.module('app.components')
- .controller('MapTagModalController', MapTagModalController);
-
- MapTagModalController.$inject = ['$mdDialog', 'tag', 'selectedTags'];
-
- function MapTagModalController($mdDialog, tag, selectedTags) {
-
- var vm = this;
-
- vm.checks = {};
-
- vm.answer = answer;
- vm.hide = hide;
- vm.clear = clear;
- vm.cancel = cancel;
- vm.tags = [];
-
- init();
-
- ////////////////////////////////////////////////////////
+ .constant('COUNTRY_CODES', {
+ 'AF': 'Afghanistan',
+ 'AX': 'Aland Islands',
+ 'AL': 'Albania',
+ 'DZ': 'Algeria',
+ 'AS': 'American Samoa',
+ 'AD': 'Andorra',
+ 'AO': 'Angola',
+ 'AI': 'Anguilla',
+ 'AQ': 'Antarctica',
+ 'AG': 'Antigua And Barbuda',
+ 'AR': 'Argentina',
+ 'AM': 'Armenia',
+ 'AW': 'Aruba',
+ 'AU': 'Australia',
+ 'AT': 'Austria',
+ 'AZ': 'Azerbaijan',
+ 'BS': 'Bahamas',
+ 'BH': 'Bahrain',
+ 'BD': 'Bangladesh',
+ 'BB': 'Barbados',
+ 'BY': 'Belarus',
+ 'BE': 'Belgium',
+ 'BZ': 'Belize',
+ 'BJ': 'Benin',
+ 'BM': 'Bermuda',
+ 'BT': 'Bhutan',
+ 'BO': 'Bolivia',
+ 'BA': 'Bosnia And Herzegovina',
+ 'BW': 'Botswana',
+ 'BV': 'Bouvet Island',
+ 'BR': 'Brazil',
+ 'IO': 'British Indian Ocean Territory',
+ 'BN': 'Brunei Darussalam',
+ 'BG': 'Bulgaria',
+ 'BF': 'Burkina Faso',
+ 'BI': 'Burundi',
+ 'KH': 'Cambodia',
+ 'CM': 'Cameroon',
+ 'CA': 'Canada',
+ 'CV': 'Cape Verde',
+ 'KY': 'Cayman Islands',
+ 'CF': 'Central African Republic',
+ 'TD': 'Chad',
+ 'CL': 'Chile',
+ 'CN': 'China',
+ 'CX': 'Christmas Island',
+ 'CC': 'Cocos (Keeling) Islands',
+ 'CO': 'Colombia',
+ 'KM': 'Comoros',
+ 'CG': 'Congo',
+ 'CD': 'Congo, Democratic Republic',
+ 'CK': 'Cook Islands',
+ 'CR': 'Costa Rica',
+ 'CI': 'Cote D\'Ivoire',
+ 'HR': 'Croatia',
+ 'CU': 'Cuba',
+ 'CY': 'Cyprus',
+ 'CZ': 'Czech Republic',
+ 'DK': 'Denmark',
+ 'DJ': 'Djibouti',
+ 'DM': 'Dominica',
+ 'DO': 'Dominican Republic',
+ 'EC': 'Ecuador',
+ 'EG': 'Egypt',
+ 'SV': 'El Salvador',
+ 'GQ': 'Equatorial Guinea',
+ 'ER': 'Eritrea',
+ 'EE': 'Estonia',
+ 'ET': 'Ethiopia',
+ 'FK': 'Falkland Islands (Malvinas)',
+ 'FO': 'Faroe Islands',
+ 'FJ': 'Fiji',
+ 'FI': 'Finland',
+ 'FR': 'France',
+ 'GF': 'French Guiana',
+ 'PF': 'French Polynesia',
+ 'TF': 'French Southern Territories',
+ 'GA': 'Gabon',
+ 'GM': 'Gambia',
+ 'GE': 'Georgia',
+ 'DE': 'Germany',
+ 'GH': 'Ghana',
+ 'GI': 'Gibraltar',
+ 'GR': 'Greece',
+ 'GL': 'Greenland',
+ 'GD': 'Grenada',
+ 'GP': 'Guadeloupe',
+ 'GU': 'Guam',
+ 'GT': 'Guatemala',
+ 'GG': 'Guernsey',
+ 'GN': 'Guinea',
+ 'GW': 'Guinea-Bissau',
+ 'GY': 'Guyana',
+ 'HT': 'Haiti',
+ 'HM': 'Heard Island & Mcdonald Islands',
+ 'VA': 'Holy See (Vatican City State)',
+ 'HN': 'Honduras',
+ 'HK': 'Hong Kong',
+ 'HU': 'Hungary',
+ 'IS': 'Iceland',
+ 'IN': 'India',
+ 'ID': 'Indonesia',
+ 'IR': 'Iran, Islamic Republic Of',
+ 'IQ': 'Iraq',
+ 'IE': 'Ireland',
+ 'IM': 'Isle Of Man',
+ 'IL': 'Israel',
+ 'IT': 'Italy',
+ 'JM': 'Jamaica',
+ 'JP': 'Japan',
+ 'JE': 'Jersey',
+ 'JO': 'Jordan',
+ 'KZ': 'Kazakhstan',
+ 'KE': 'Kenya',
+ 'KI': 'Kiribati',
+ 'KR': 'Korea',
+ 'KW': 'Kuwait',
+ 'KG': 'Kyrgyzstan',
+ 'LA': 'Lao People\'s Democratic Republic',
+ 'LV': 'Latvia',
+ 'LB': 'Lebanon',
+ 'LS': 'Lesotho',
+ 'LR': 'Liberia',
+ 'LY': 'Libyan Arab Jamahiriya',
+ 'LI': 'Liechtenstein',
+ 'LT': 'Lithuania',
+ 'LU': 'Luxembourg',
+ 'MO': 'Macao',
+ 'MK': 'Macedonia',
+ 'MG': 'Madagascar',
+ 'MW': 'Malawi',
+ 'MY': 'Malaysia',
+ 'MV': 'Maldives',
+ 'ML': 'Mali',
+ 'MT': 'Malta',
+ 'MH': 'Marshall Islands',
+ 'MQ': 'Martinique',
+ 'MR': 'Mauritania',
+ 'MU': 'Mauritius',
+ 'YT': 'Mayotte',
+ 'MX': 'Mexico',
+ 'FM': 'Micronesia, Federated States Of',
+ 'MD': 'Moldova',
+ 'MC': 'Monaco',
+ 'MN': 'Mongolia',
+ 'ME': 'Montenegro',
+ 'MS': 'Montserrat',
+ 'MA': 'Morocco',
+ 'MZ': 'Mozambique',
+ 'MM': 'Myanmar',
+ 'NA': 'Namibia',
+ 'NR': 'Nauru',
+ 'NP': 'Nepal',
+ 'NL': 'Netherlands',
+ 'AN': 'Netherlands Antilles',
+ 'NC': 'New Caledonia',
+ 'NZ': 'New Zealand',
+ 'NI': 'Nicaragua',
+ 'NE': 'Niger',
+ 'NG': 'Nigeria',
+ 'NU': 'Niue',
+ 'NF': 'Norfolk Island',
+ 'MP': 'Northern Mariana Islands',
+ 'NO': 'Norway',
+ 'OM': 'Oman',
+ 'PK': 'Pakistan',
+ 'PW': 'Palau',
+ 'PS': 'Palestinian Territory, Occupied',
+ 'PA': 'Panama',
+ 'PG': 'Papua New Guinea',
+ 'PY': 'Paraguay',
+ 'PE': 'Peru',
+ 'PH': 'Philippines',
+ 'PN': 'Pitcairn',
+ 'PL': 'Poland',
+ 'PT': 'Portugal',
+ 'PR': 'Puerto Rico',
+ 'QA': 'Qatar',
+ 'RE': 'Reunion',
+ 'RO': 'Romania',
+ 'RU': 'Russian Federation',
+ 'RW': 'Rwanda',
+ 'BL': 'Saint Barthelemy',
+ 'SH': 'Saint Helena',
+ 'KN': 'Saint Kitts And Nevis',
+ 'LC': 'Saint Lucia',
+ 'MF': 'Saint Martin',
+ 'PM': 'Saint Pierre And Miquelon',
+ 'VC': 'Saint Vincent And Grenadines',
+ 'WS': 'Samoa',
+ 'SM': 'San Marino',
+ 'ST': 'Sao Tome And Principe',
+ 'SA': 'Saudi Arabia',
+ 'SN': 'Senegal',
+ 'RS': 'Serbia',
+ 'SC': 'Seychelles',
+ 'SL': 'Sierra Leone',
+ 'SG': 'Singapore',
+ 'SK': 'Slovakia',
+ 'SI': 'Slovenia',
+ 'SB': 'Solomon Islands',
+ 'SO': 'Somalia',
+ 'ZA': 'South Africa',
+ 'GS': 'South Georgia And Sandwich Isl.',
+ 'ES': 'Spain',
+ 'LK': 'Sri Lanka',
+ 'SD': 'Sudan',
+ 'SR': 'Suriname',
+ 'SJ': 'Svalbard And Jan Mayen',
+ 'SZ': 'Swaziland',
+ 'SE': 'Sweden',
+ 'CH': 'Switzerland',
+ 'SY': 'Syrian Arab Republic',
+ 'TW': 'Taiwan',
+ 'TJ': 'Tajikistan',
+ 'TZ': 'Tanzania',
+ 'TH': 'Thailand',
+ 'TL': 'Timor-Leste',
+ 'TG': 'Togo',
+ 'TK': 'Tokelau',
+ 'TO': 'Tonga',
+ 'TT': 'Trinidad And Tobago',
+ 'TN': 'Tunisia',
+ 'TR': 'Turkey',
+ 'TM': 'Turkmenistan',
+ 'TC': 'Turks And Caicos Islands',
+ 'TV': 'Tuvalu',
+ 'UG': 'Uganda',
+ 'UA': 'Ukraine',
+ 'AE': 'United Arab Emirates',
+ 'GB': 'United Kingdom',
+ 'US': 'United States',
+ 'UM': 'United States Outlying Islands',
+ 'UY': 'Uruguay',
+ 'UZ': 'Uzbekistan',
+ 'VU': 'Vanuatu',
+ 'VE': 'Venezuela',
+ 'VN': 'Viet Nam',
+ 'VG': 'Virgin Islands, British',
+ 'VI': 'Virgin Islands, U.S.',
+ 'WF': 'Wallis And Futuna',
+ 'EH': 'Western Sahara',
+ 'YE': 'Yemen',
+ 'ZM': 'Zambia',
+ 'ZW': 'Zimbabwe'
+ });
+})();
- function init() {
- tag.getTags()
- .then(function(tags) {
- vm.tags = tags;
+(function() {
+ 'use strict';
- _.forEach(selectedTags, select);
+ angular.module('app.components')
+ .factory('user', user);
- });
- }
+ user.$inject = ['Restangular'];
+ function user(Restangular) {
+ var service = {
+ createUser: createUser,
+ getUser: getUser,
+ updateUser: updateUser
+ };
+ return service;
- function answer() {
+ ////////////////////
- var selectedTags = _(vm.tags)
- .filter(isTagSelected)
- .value();
- $mdDialog.hide(selectedTags);
- }
+ function createUser(signupData) {
+ return Restangular.all('users').post(signupData);
+ }
- function hide() {
- answer();
- }
+ function getUser(id) {
+ return Restangular.one('users', id).get();
+ }
- function clear() {
- $mdDialog.hide(null);
- }
+ function updateUser(updateData) {
+ return Restangular.all('me').customPUT(updateData);
+ }
+ }
+})();
- function cancel() {
- answer();
- }
+(function() {
+ 'use strict';
- function isTagSelected(tag) {
- return vm.checks[tag.name];
- }
+ angular.module('app.components')
+ .factory('tag', tag);
- function select(tag){
- vm.checks[tag] = true;
+ tag.$inject = ['Restangular'];
+ function tag(Restangular) {
+ var tags = [];
+ var selectedTags = [];
+
+ var service = {
+ getTags: getTags,
+ getSelectedTags: getSelectedTags,
+ setSelectedTags: setSelectedTags,
+ tagWithName: tagWithName,
+ filterMarkersByTag: filterMarkersByTag
+ };
+
+ return service;
+
+ /////////////////
+
+ function getTags() {
+ return Restangular.all('tags')
+ .getList({'per_page': 200})
+ .then(function(fetchedTags){
+ tags = fetchedTags.plain();
+ return tags;
+ });
+ }
+
+ function getSelectedTags(){
+ return selectedTags;
+ }
+
+ function setSelectedTags(tags){
+ selectedTags = tags;
+ }
+
+ function tagWithName(name){
+ var result = _.where(tags, {name: name});
+ if (result && result.length > 0){
+ return result[0];
+ }else{
+ return;
+ }
+ }
+
+ function filterMarkersByTag(tmpMarkers) {
+ var markers = filterMarkers(tmpMarkers);
+ return markers;
+ }
+
+ function filterMarkers(tmpMarkers) {
+ if (service.getSelectedTags().length === 0){
+ return tmpMarkers;
+ }
+ return tmpMarkers.filter(function(marker) {
+ var tags = marker.myData.tags;
+ if (tags.length === 0){
+ return false;
+ }
+ return _.some(tags, function(tag) {
+ return _.includes(service.getSelectedTags(), tag);
+ });
+ });
+ }
}
- }
})();
(function() {
'use strict';
angular.module('app.components')
- .controller('MapFilterModalController', MapFilterModalController);
+ .factory('sensor', sensor);
- MapFilterModalController.$inject = ['$mdDialog','selectedFilters', '$timeout'];
+ sensor.$inject = ['Restangular', 'timeUtils', 'sensorUtils'];
+ function sensor(Restangular, timeUtils, sensorUtils) {
+ var sensorTypes;
+ callAPI().then(function(data) {
+ setTypes(data);
+ });
- function MapFilterModalController($mdDialog, selectedFilters, $timeout) {
+ var service = {
+ callAPI: callAPI,
+ setTypes: setTypes,
+ getTypes: getTypes,
+ getSensorsData: getSensorsData
+ };
+ return service;
- var vm = this;
+ ////////////////
- vm.checks = {};
+ function callAPI() {
+ return Restangular.all('sensors').getList({'per_page': 1000});
+ }
- vm.answer = answer;
- vm.hide = hide;
- vm.clear = clear;
- vm.cancel = cancel;
- vm.toggle = toggle;
+ function setTypes(sensorTypes) {
+ sensorTypes = sensorTypes;
+ }
- vm.location = ['indoor', 'outdoor'];
- vm.status = ['online', 'offline'];
- vm.new = ['new'];
+ function getTypes() {
+ return sensorTypes;
+ }
- vm.filters = [];
+ function getSensorsData(deviceID, sensorID, dateFrom, dateTo) {
+ var rollup = sensorUtils.getRollup(dateFrom, dateTo);
+ dateFrom = timeUtils.convertTime(dateFrom);
+ dateTo = timeUtils.convertTime(dateTo);
- init();
+ return Restangular.one('devices', deviceID).customGET('readings', {'from': dateFrom, 'to': dateTo, 'rollup': rollup, 'sensor_id': sensorID, 'all_intervals': true});
+ }
+ }
+})();
- ////////////////////////////////////////////////////////
+(function() {
+ 'use strict';
- function init() {
- _.forEach(selectedFilters, select);
- }
+ angular.module('app.components')
+ .factory('search', search);
+
+ search.$inject = ['$http', 'Restangular'];
+ function search($http, Restangular) {
+ var service = {
+ globalSearch: globalSearch
+ };
- function answer() {
- vm.filters = vm.filters.concat(vm.location, vm.status, vm.new);
- var selectedFilters = _(vm.filters)
- .filter(isFilterSelected)
- .value();
- $mdDialog.hide(selectedFilters);
- }
+ return service;
- function hide() {
- answer();
- }
+ /////////////////////////
- function clear() {
- vm.filters = vm.filters.concat(vm.location, vm.status, vm.new);
- $mdDialog.hide(vm.filters);
+ function globalSearch(query) {
+ return Restangular.all('search').getList({q: query});
+ }
}
+})();
- function cancel() {
- answer();
- }
+(function() {
+ 'use strict';
- function isFilterSelected(filter) {
- return vm.checks[filter];
- }
+ angular.module('app.components')
+ .factory('measurement', measurement);
- function toggle(filters) {
- $timeout(function() {
+ measurement.$inject = ['Restangular'];
- for (var i = 0; i < filters.length - 1; i++) {
- if (vm.checks[filters[i]] === false && vm.checks[filters[i]] === vm.checks[filters[i+1]]) {
- for (var n = 0; n < filters.length; n++) {
- vm.checks[filters[n]] = true;
- }
- }
- }
+ function measurement(Restangular) {
- });
+ var service = {
+ getTypes: getTypes,
+ getMeasurement: getMeasurement
+
+ };
+ return service;
+
+ ////////////////
+
+
+ function getTypes() {
+ return Restangular.all('measurements').getList({'per_page': 1000});
}
- function select(filter){
- vm.checks[filter] = true;
+ function getMeasurement(mesID) {
+
+ return Restangular.one('measurements', mesID).get();
}
}
})();
-
(function() {
- 'use strict';
+ 'use strict';
- angular.module('app.components')
- .controller('MapController', MapController);
+ angular.module('app.components')
+ .factory('geolocation', geolocation);
- MapController.$inject = ['$scope', '$state', '$stateParams', '$timeout', 'device',
- '$mdDialog', 'leafletData', 'alert',
- 'Marker', 'tag', 'animation', '$q'];
- function MapController($scope, $state, $stateParams, $timeout, device,
- $mdDialog, leafletData, alert, Marker, tag, animation, $q) {
- var vm = this;
- var updateType;
- var focusedMarkerID;
+ geolocation.$inject = ['$http', '$window'];
+ function geolocation($http, $window) {
- vm.markers = [];
+ var service = {
+ grantHTML5Geolocation: grantHTML5Geolocation,
+ isHTML5GeolocationGranted: isHTML5GeolocationGranted
+ };
+ return service;
- var retinaSuffix = isRetina() ? '512' : '256';
- var retinaLegacySuffix = isRetina() ? '@2x' : '';
-
- var mapBoxToken = 'pk.eyJ1IjoidG9tYXNkaWV6IiwiYSI6ImRTd01HSGsifQ.loQdtLNQ8GJkJl2LUzzxVg';
-
- vm.layers = {
- baselayers: {
- osm: {
- name: 'OpenStreetMap',
- type: 'xyz',
- url: 'https://api.mapbox.com/styles/v1/mapbox/streets-v10/tiles/' + retinaSuffix + '/{z}/{x}/{y}?access_token=' + mapBoxToken
- },
- legacy: {
- name: 'Legacy',
- type: 'xyz',
- url: 'https://api.tiles.mapbox.com/v4/mapbox.streets-basic/{z}/{x}/{y}'+ retinaLegacySuffix +'.png' + '?access_token=' + mapBoxToken
- },
- sat: {
- name: 'Satellite',
- type: 'xyz',
- url: 'https://api.mapbox.com/styles/v1/mapbox/satellite-streets-v10/tiles/' + retinaSuffix + '/{z}/{x}/{y}?access_token=' + mapBoxToken
- }
- },
- overlays: {
- devices: {
- name: 'Devices',
- type: 'markercluster',
- visible: true,
- layerOptions: {
- showCoverageOnHover: false
- }
- }
- }
- };
-
- vm.center = {
- lat: $stateParams.lat ? parseInt($stateParams.lat, 10) : 13.14950321154457,
- lng: $stateParams.lng ? parseInt($stateParams.lng, 10) : -1.58203125,
- zoom: $stateParams.zoom ? parseInt($stateParams.zoom, 10) : 2
- };
-
- vm.defaults = {
- dragging: true,
- touchZoom: true,
- scrollWheelZoom: true,
- doubleClickZoom: true,
- minZoom:2,
- worldCopyJump: true
- };
-
- vm.events = {
- map: {
- enable: ['dragend', 'zoomend', 'moveend', 'popupopen', 'popupclose',
- 'mousedown', 'dblclick', 'click', 'touchstart', 'mouseup'],
- logic: 'broadcast'
- }
- };
+ ///////////////////////////
- $scope.$on('leafletDirectiveMarker.click', function(event, data) {
- var id = undefined;
- var currentMarker = vm.markers[data.modelName];
- if(currentMarker) {
- id = currentMarker.myData.id;
- }
+ function grantHTML5Geolocation(){
+ $window.localStorage.setItem('smartcitizen.geolocation_granted', true);
+ }
- vm.deviceLoading = true;
- vm.center.lat = data.leafletEvent.latlng.lat;
- vm.center.lng = data.leafletEvent.latlng.lng;
+ function isHTML5GeolocationGranted(){
+ return $window.localStorage
+ .getItem('smartcitizen.geolocation_granted');
+ }
+ }
+})();
- if(id === parseInt($state.params.id)) {
- $timeout(function() {
- vm.deviceLoading = false;
- });
- return;
- }
+(function() {
+ 'use strict';
- updateType = 'map';
+ angular.module('app.components')
+ .factory('file', file);
- if ($state.$current.name === 'embbed') { return; }
- $state.go('layout.home.kit', {id: id});
+ file.$inject = ['Restangular', 'Upload'];
+ function file(Restangular, Upload) {
+ var service = {
+ getCredentials: getCredentials,
+ uploadFile: uploadFile,
+ getImageURL: getImageURL
+ };
+ return service;
- // angular.element('section.map').scope().$broadcast('resizeMapHeight');
- });
+ ///////////////
+ function getCredentials(filename) {
+ var data = {
+ filename: filename
+ };
+ return Restangular.all('me/avatar').post(data);
+ }
- $scope.$on('leafletDirectiveMarker.popupclose', function() {
- if(focusedMarkerID) {
- var marker = vm.markers[focusedMarkerID];
- if(marker) {
- vm.markers[focusedMarkerID].focus = false;
+ function uploadFile(fileData, key, policy, signature) {
+ return Upload.upload({
+ url: 'https://smartcitizen.s3-eu-west-1.amazonaws.com',
+ method: 'POST',
+ data: {
+ key: key,
+ policy: policy,
+ signature: signature,
+ AWSAccessKeyId: 'AKIAJ753OQI6JPSDCPHA',
+ acl: 'public-read',
+ "Content-Type": fileData.type || 'application/octet-stream',
+ /*jshint camelcase: false */
+ success_action_status: 200,
+ file: fileData
}
- }
- });
+ });
+ }
- vm.readyForDevice = {
- device: false,
- map: false
- };
+ function getImageURL(filename, size) {
+ size = size === undefined ? 's101' : size;
- $scope.$on('deviceLoaded', function(event, data) {
- vm.readyForDevice.device = data;
- });
+ return 'https://images.smartcitizen.me/' + size + '/' + filename;
+ }
+ }
+})();
- $scope.$watch('vm.readyForDevice', function() {
- if (vm.readyForDevice.device && vm.readyForDevice.map) {
- zoomDeviceAndPopUp(vm.readyForDevice.device);
- }
- }, true);
+(function() {
+ 'use strict';
- $scope.$on('goToLocation', function(event, data) {
- goToLocation(data);
- });
+ angular.module('app.components')
+ .factory('device', device);
- vm.filters = ['indoor', 'outdoor', 'online', 'offline'];
+ device.$inject = ['Restangular', '$window', 'timeUtils','$http', 'auth', '$rootScope'];
+ function device(Restangular, $window, timeUtils, $http, auth, $rootScope) {
+ var worldMarkers;
- vm.openFilterPopup = openFilterPopup;
- vm.openTagPopup = openTagPopup;
- vm.removeFilter = removeFilter;
- vm.removeTag = removeTag;
- vm.selectedTags = tag.getSelectedTags();
- vm.selectedFilters = ['indoor', 'outdoor', 'online', 'offline', 'new'];
+ initialize();
- vm.checkAllFiltersSelected = checkAllFiltersSelected;
+ var service = {
+ getDevices: getDevices,
+ getAllDevices: getAllDevices,
+ getDevice: getDevice,
+ createDevice: createDevice,
+ updateDevice: updateDevice,
+ getWorldMarkers: getWorldMarkers,
+ setWorldMarkers: setWorldMarkers,
+ mailReadings: mailReadings,
+ postReadings: postReadings,
+ removeDevice: removeDevice,
+ updateContext: updateContext
+ };
- initialize();
+ return service;
- /////////////////////
+ //////////////////////////
function initialize() {
+ if(areMarkersOld()) {
+ removeMarkers();
+ }
+ }
- vm.readyForDevice.map = false;
+ function getDevices(location) {
+ var parameter = '';
+ parameter += location.lat + ',' + location.lng;
+ return Restangular.all('devices').getList({near: parameter, 'per_page': '100'});
+ }
- $q.all([device.getAllDevices($stateParams.reloadMap)])
- .then(function(data){
+ function getAllDevices(forceReload) {
+ if (forceReload || auth.isAuth()) {
+ return getAllDevicesNoCached();
+ } else {
+ return getAllDevicesCached();
+ }
+ }
- data = data[0];
+ function getAllDevicesCached() {
+ return Restangular.all('devices/world_map')
+ .getList()
+ .then(function(fetchedDevices){
+ return fetchedDevices.plain();
+ });
+ }
- vm.markers = _.chain(data)
- .map(function(device) {
- return new Marker(device);
- })
- .filter(function(marker) {
- return !!marker.lng && !!marker.lat;
- })
- .tap(function(data) {
- device.setWorldMarkers(data);
- })
- .value();
+ function getAllDevicesNoCached() {
+ return Restangular.all('devices/fresh_world_map')
+ .getList()
+ .then(function(fetchedDevices){
+ return fetchedDevices.plain();
+ });
+ }
- var markersByIndex = _.keyBy(vm.markers, function(marker) {
- return marker.myData.id;
- });
+ function getDevice(id) {
+ return Restangular.one('devices', id).get();
+ }
- if($state.params.id && markersByIndex[parseInt($state.params.id)]){
- focusedMarkerID = markersByIndex[parseInt($state.params.id)]
- .myData.id;
- vm.readyForDevice.map = true;
- } else {
- updateMarkers();
- vm.readyForDevice.map = true;
- }
+ function createDevice(data) {
+ return Restangular.all('devices').post(data);
+ }
- });
+ function updateDevice(id, data) {
+ return Restangular.one('devices', id).patch(data);
}
- function zoomDeviceAndPopUp(data){
+ function getWorldMarkers() {
+ return worldMarkers || ($window.localStorage.getItem('smartcitizen.markers') && JSON.parse($window.localStorage.getItem('smartcitizen.markers') ).data);
+ }
- if(updateType === 'map') {
- vm.deviceLoading = false;
- updateType = undefined;
- return;
- } else {
- vm.deviceLoading = true;
+ function setWorldMarkers(data) {
+ var obj = {
+ timestamp: new Date(),
+ data: data
+ };
+ try {
+ $window.localStorage.setItem('smartcitizen.markers', JSON.stringify(obj) );
+ } catch (e) {
+ console.log("Could not store markers in localstorage. skipping...");
}
-
- leafletData.getMarkers()
- .then(function(markers) {
- var currentMarker = _.find(markers, function(marker) {
- return data.id === marker.options.myData.id;
- });
-
- var id = data.id;
-
- leafletData.getLayers()
- .then(function(layers) {
- if(currentMarker){
- layers.overlays.devices.zoomToShowLayer(currentMarker,
- function() {
- var selectedMarker = currentMarker;
- if(selectedMarker) {
- // Ensures the marker is not just zoomed but the marker is centered to improve UX
- // The $timeout can be replaced by an event but tests didn't show good results
- $timeout(function() {
- vm.center.lat = selectedMarker.options.lat;
- vm.center.lng = selectedMarker.options.lng;
- selectedMarker.openPopup();
- vm.deviceLoading = false;
- }, 1000);
- }
- });
- } else {
- leafletData.getMap().then(function(map){
- map.closePopup();
- });
- }
- });
- });
-
+ worldMarkers = obj.data;
}
- function checkAllFiltersSelected() {
- var allFiltersSelected = _.every(vm.filters, function(filterValue) {
- return _.includes(vm.selectedFilters, filterValue);
- });
- return allFiltersSelected;
+ function getTimeStamp() {
+ return ($window.localStorage.getItem('smartcitizen.markers') &&
+ JSON.parse($window.localStorage
+ .getItem('smartcitizen.markers') ).timestamp);
}
- function openFilterPopup() {
- $mdDialog.show({
- hasBackdrop: true,
- controller: 'MapFilterModalController',
- controllerAs: 'vm',
- templateUrl: 'app/components/map/mapFilterModal.html',
- clickOutsideToClose: true,
- locals: {
- selectedFilters: vm.selectedFilters
- }
- })
- .then(function(selectedFilters) {
- updateType = 'map';
- vm.selectedFilters = selectedFilters;
- updateMapFilters();
- });
+ function areMarkersOld() {
+ var markersDate = getTimeStamp();
+ return !timeUtils.isWithin(1, 'minutes', markersDate);
}
- function openTagPopup() {
- $mdDialog.show({
- hasBackdrop: true,
- controller: 'MapTagModalController',
- controllerAs: 'vm',
- templateUrl: 'app/components/map/mapTagModal.html',
- //targetEvent: ev,
- clickOutsideToClose: true,
- locals: {
- selectedTags: vm.selectedTags
- }
- })
- .then(function(selectedTags) {
- if (selectedTags && selectedTags.length > 0) {
- updateType = 'map';
- tag.setSelectedTags(_.map(selectedTags, 'name'));
- vm.selectedTags = tag.getSelectedTags();
- reloadWithTags();
- } else if (selectedTags === null) {
- reloadNoTags();
- }
- });
+ function removeMarkers() {
+ worldMarkers = null;
+ $window.localStorage.removeItem('smartcitizen.markers');
}
- function updateMapFilters(){
- vm.selectedTags = tag.getSelectedTags();
- checkAllFiltersSelected();
- updateMarkers();
+ function mailReadings(kit) {
+ return Restangular
+ .one('devices', kit.id)
+ .customGET('readings/csv_archive');
}
- function removeFilter(filterName) {
- vm.selectedFilters = _.filter(vm.selectedFilters, function(el){
- return el !== filterName;
- });
- if(vm.selectedFilters.length === 0){
- vm.selectedFilters = vm.filters;
- }
- updateMarkers();
- }
+ function postReadings(kit, readings) {
+ return Restangular
+ .one('devices', kit.id)
+ .post('readings', readings);
+ }
- function filterMarkersByLabel(tmpMarkers) {
- return tmpMarkers.filter(function(marker) {
- var labels = marker.myData.labels;
- if (labels.length === 0 && vm.selectedFilters.length !== 0){
- return false;
- }
- return _.every(labels, function(label) {
- return _.includes(vm.selectedFilters, label);
- });
+ function removeDevice(deviceID){
+ return Restangular
+ .one('devices', deviceID)
+ .remove().then(function () {
+ $rootScope.$broadcast('devicesContextUpdated');
+ })
+ ;
+ }
+
+ function updateContext (){
+ return auth.updateUser().then(function(){
+ removeMarkers();
+ $rootScope.$broadcast('devicesContextUpdated');
});
}
- function updateMarkers() {
- $timeout(function() {
- $scope.$apply(function() {
- var allMarkers = device.getWorldMarkers();
+ }
+})();
- var updatedMarkers = allMarkers;
+(function() {
+ 'use strict';
- updatedMarkers = tag.filterMarkersByTag(updatedMarkers);
- updatedMarkers = filterMarkersByLabel(updatedMarkers);
- vm.markers = updatedMarkers;
+ angular.module('app.components')
+ .factory('auth', auth);
- animation.mapStateLoaded();
+ auth.$inject = ['$location', '$window', '$state', 'Restangular',
+ '$rootScope', 'AuthUser', '$timeout', 'alert', '$cookies'];
+ function auth($location, $window, $state, Restangular, $rootScope, AuthUser,
+ $timeout, alert, $cookies) {
- vm.deviceLoading = false;
+ var user = {};
- zoomOnMarkers();
- });
- });
- }
+ //wait until http interceptor is added to Restangular
+ $timeout(function() {
+ initialize();
+ }, 100);
- function getZoomLevel(data) {
- // data.layer is an array of strings like ["establishment", "point_of_interest"]
- var zoom = 18;
+ var service = {
+ isAuth: isAuth,
+ setCurrentUser: setCurrentUser,
+ getCurrentUser: getCurrentUser,
+ updateUser: updateUser,
+ saveToken: saveToken,
+ getToken: getToken,
+ login: login,
+ logout: logout,
+ recoverPassword: recoverPassword,
+ getResetPassword: getResetPassword,
+ patchResetPassword: patchResetPassword,
+ isAdmin: isAdmin
+ };
+ return service;
- if(data.layer && data.layer[0]) {
- switch(data.layer[0]) {
- case 'point_of_interest':
- zoom = 18;
- break;
- case 'address':
- zoom = 18;
- break;
- case "establishment":
- zoom = 15;
- break;
- case 'neighbourhood':
- zoom = 13;
- break;
- case 'locality':
- zoom = 13;
- break;
- case 'localadmin':
- zoom = 9;
- break;
- case 'county':
- zoom = 9;
- break;
- case 'region':
- zoom = 8;
- break;
- case 'country':
- zoom = 7;
- break;
- case 'coarse':
- zoom = 7;
- break;
- }
- }
+ //////////////////////////
- return zoom;
+ function initialize() {
+ //console.log('---- AUTH INIT -----');
+ setCurrentUser('appLoad');
}
- function isRetina(){
- return ((window.matchMedia &&
- (window.matchMedia('only screen and (min-resolution: 192dpi), ' +
- 'only screen and (min-resolution: 2dppx), only screen and ' +
- '(min-resolution: 75.6dpcm)').matches ||
- window.matchMedia('only screen and (-webkit-min-device-pixel-ra' +
- 'tio: 2), only screen and (-o-min-device-pixel-ratio: 2/1), only' +
- ' screen and (min--moz-device-pixel-ratio: 2), only screen and ' +
- '(min-device-pixel-ratio: 2)').matches)) ||
- (window.devicePixelRatio && window.devicePixelRatio >= 2)) &&
- /(iPad|iPhone|iPod|Apple)/g.test(navigator.userAgent);
- }
+ //run on app initialization so that we can keep auth across different sessions
+ // 1. Check if token in cookie exists. Return if it doesn't, user needs to login (and save a token to the cookie)
+ // 2. Populate user.data with the response from the API.
+ // 3. Broadcast logged in
+ function setCurrentUser(time) {
+ // TODO later: Should we check if token is expired here?
+ if (getToken()) {
+ user.token = getToken();
+ }else{
+ //console.log('token not found in cookie, returning');
+ return;
+ }
- function goToLocation(data){
- // This ensures the action runs after the event is registered
- $timeout(function() {
- vm.center.lat = data.lat;
- vm.center.lng = data.lng;
- vm.center.zoom = getZoomLevel(data);
- });
+ return getCurrentUserFromAPI()
+ .then(function(data) {
+ // Save user.data also in localStorage. It is beeing used across the app.
+ // Should it instead just be saved in the user object? Or is it OK to also have it in localStorage?
+ $window.localStorage.setItem('smartcitizen.data', JSON.stringify(data.plain()) );
+
+ var newUser = new AuthUser(data);
+ //check sensitive information
+ if(user.data && user.data.role !== newUser.role) {
+ user.data = newUser;
+ $location.path('/');
+ }
+ user.data = newUser;
+
+ //console.log('-- User populated with data: ', user)
+ // Broadcast happens 2x, so the user wont think he is not logged in.
+ // The 2nd broadcast waits 3sec, because f.x. on the /kits/ page, the layout has not loaded when the broadcast is sent
+ $rootScope.$broadcast('loggedIn');
+
+ // used for app initialization
+ if(time && time === 'appLoad') {
+ //wait until navbar is loaded to emit event
+ $timeout(function() {
+ $rootScope.$broadcast('loggedIn', {time: 'appLoad'});
+ }, 3000);
+ } else {
+ // used for login
+ //$state.reload();
+ $timeout(function() {
+ alert.success('Login was successful');
+ $rootScope.$broadcast('loggedIn', {});
+ }, 2000);
+ }
+ });
}
- function removeTag(tagName){
- tag.setSelectedTags(_.filter(vm.selectedTags, function(el){
- return el !== tagName;
- }));
+ // Called from device.service.js updateContext(), which is called from multiple /kit/ pages
+ function updateUser() {
+ return getCurrentUserFromAPI()
+ .then(function(data) {
+ // TODO: Should this update the token or user.data? Then it could instead call setCurrentUser?
+ $window.localStorage.setItem('smartcitizen.data', JSON.stringify(data.plain()) );
+ return getCurrentUser();
+ });
+ }
- vm.selectedTags = tag.getSelectedTags();
+ function getCurrentUser() {
+ user.token = getToken();
+ user.data = $window.localStorage.getItem('smartcitizen.data') && new AuthUser(JSON.parse( $window.localStorage.getItem('smartcitizen.data') ));
+ return user;
+ }
- if(vm.selectedTags.length === 0){
- reloadNoTags();
- } else {
- reloadWithTags();
- }
+ // Should check if user.token exists - but now checks if the cookies.token exists.
+ function isAuth() {
+ // TODO: isAuth() is called from many different services BEFORE auth.init has run.
+ // That means that the user.token is EMPTY, meaning isAuth will be false
+ // We can cheat and just check the cookie, but we should NOT. Because auth.init should also check if the cookie is valid / expired
+ // Ideally it should return !!user.token
+ //return !!user.token;
+ return !!getToken();
+ }
+ // LoginModal calls this after it receives the token from the API, and wants to save it in a cookie.
+ function saveToken(token) {
+ //console.log('saving Token to cookie:', token);
+ $cookies.put('smartcitizen.token', token);
+ setCurrentUser();
}
- function zoomOnMarkers(){
- $timeout(function() {
- if(vm.markers && vm.markers.length > 0) {
- leafletData.getMap().then(function(map){
- var bounds = L.latLngBounds(vm.markers);
- map.fitBounds(bounds);
- });
- } else {
- alert.error('No markers found with those filters', 5000);
- }
- });
+ function getToken(){
+ return $cookies.get('smartcitizen.token');
}
- function reloadWithTags(){
- $state.transitionTo('layout.home.tags', {tags: vm.selectedTags}, {reload: true});
+ function login(loginData) {
+ return Restangular.all('sessions').post(loginData);
}
- function reloadNoTags(){
- $state.transitionTo('layout.home.kit');
+ function logout() {
+ $cookies.remove('smartcitizen.token');
}
- }
+ function getCurrentUserFromAPI() {
+ return Restangular.all('').customGET('me');
+ }
+
+ function recoverPassword(data) {
+ return Restangular.all('password_resets').post(data);
+ }
+ function getResetPassword(code) {
+ return Restangular.one('password_resets', code).get();
+ }
+ function patchResetPassword(code, data) {
+ return Restangular.one('password_resets', code).patch(data);
+ }
+ function isAdmin(userData) {
+ return userData.role === 'admin';
+ }
+ }
})();
(function() {
'use strict';
+ /**
+ * Unused directive. Double-check before removing.
+ *
+ */
angular.module('app.components')
- .controller('LoginModalController', LoginModalController);
+ .directive('slide', slide)
+ .directive('slideMenu', slideMenu);
- LoginModalController.$inject = ['$scope', '$mdDialog', 'auth', 'animation'];
- function LoginModalController($scope, $mdDialog, auth, animation) {
- const vm = this;
- $scope.answer = function(answer) {
- $scope.waitingFromServer = true;
- auth.login(answer)
- .then(function(data) {
- /*jshint camelcase: false */
- var token = data.access_token;
- auth.saveToken(token);
- $mdDialog.hide();
- })
- .catch(function(err) {
- vm.errors = err.data;
- })
- .finally(function() {
- $scope.waitingFromServer = false;
- });
- };
- $scope.hide = function() {
- $mdDialog.hide();
- };
- $scope.cancel = function() {
- $mdDialog.hide();
+ function slideMenu() {
+ return {
+ controller: controller,
+ link: link
};
- $scope.openSignup = function() {
- animation.showSignup();
- $mdDialog.hide();
- };
+ function link(scope, element) {
+ scope.element = element;
+ }
- $scope.openPasswordRecovery = function() {
- $mdDialog.show({
- hasBackdrop: true,
- controller: 'PasswordRecoveryModalController',
- templateUrl: 'app/components/passwordRecovery/passwordRecoveryModal.html',
- clickOutsideToClose: true
- });
+ function controller($scope) {
+ $scope.slidePosition = 0;
+ $scope.slideSize = 20;
- $mdDialog.hide();
+ this.getTimesSlided = function() {
+ return $scope.slideSize;
+ };
+ this.getPosition = function() {
+ return $scope.slidePosition * $scope.slideSize;
+ };
+ this.decrementPosition = function() {
+ $scope.slidePosition -= 1;
+ };
+ this.incrementPosition = function() {
+ $scope.slidePosition += 1;
+ };
+ this.scrollIsValid = function(direction) {
+ var scrollPosition = $scope.element.scrollLeft();
+ console.log('scrollpos', scrollPosition);
+ if(direction === 'left') {
+ return scrollPosition > 0 && $scope.slidePosition >= 0;
+ } else if(direction === 'right') {
+ return scrollPosition < 300;
+ }
+ };
+ }
+ }
+
+ slide.$inject = [];
+ function slide() {
+ return {
+ link: link,
+ require: '^slide-menu',
+ restrict: 'A',
+ scope: {
+ direction: '@'
+ }
};
+
+ function link(scope, element, attr, slideMenuCtrl) {
+ //select first sensor container
+ var sensorsContainer = angular.element('.sensors_container');
+
+ element.on('click', function() {
+
+ if(slideMenuCtrl.scrollIsValid('left') && attr.direction === 'left') {
+ slideMenuCtrl.decrementPosition();
+ sensorsContainer.scrollLeft(slideMenuCtrl.getPosition());
+ console.log(slideMenuCtrl.getPosition());
+ } else if(slideMenuCtrl.scrollIsValid('right') && attr.direction === 'right') {
+ slideMenuCtrl.incrementPosition();
+ sensorsContainer.scrollLeft(slideMenuCtrl.getPosition());
+ console.log(slideMenuCtrl.getPosition());
+ }
+ });
+ }
}
})();
(function() {
'use strict';
- angular.module('app.components')
- .directive('login', login);
+ angular.module('app.components')
+ .directive('showPopupInfo', showPopupInfo);
- function login() {
+ /**
+ * Used to show/hide explanation of sensor value at kit dashboard
+ *
+ */
+ showPopupInfo.$inject = [];
+ function showPopupInfo() {
return {
- scope: {
- show: '='
- },
- restrict: 'A',
- controller: 'LoginController',
- controllerAs: 'vm',
- templateUrl: 'app/components/login/login.html'
+ link: link
};
+
+ //////
+
+
+ function link(scope, elem) {
+ elem.on('mouseenter', function() {
+ angular.element('.sensor_data_description').css('display', 'inline-block');
+ });
+ elem.on('mouseleave', function() {
+ angular.element('.sensor_data_description').css('display', 'none');
+ });
+ }
}
})();
@@ -3715,131 +3735,90 @@ angular.module('app.components')
'use strict';
angular.module('app.components')
- .controller('LoginController', LoginController);
+ .directive('showPopup', showPopup);
- LoginController.$inject = ['$scope', '$mdDialog'];
- function LoginController($scope, $mdDialog) {
+ /**
+ * Used on kit dashboard to open full sensor description
+ */
- $scope.showLogin = showLogin;
-
- $scope.$on('showLogin', function() {
- showLogin();
- });
+ showPopup.$inject = [];
+ function showPopup() {
+ return {
+ link: link
+ };
- ////////////////
+ /////
- function showLogin() {
- $mdDialog.show({
- hasBackdrop: true,
- fullscreen: true,
- controller: 'LoginModalController',
- controllerAs: 'vm',
- templateUrl: 'app/components/login/loginModal.html',
- clickOutsideToClose: true
- });
+ function link(scope, element) {
+ element.on('click', function() {
+ var text = angular.element('.sensor_description_preview').text();
+ if(text.length < 140) {
+ return;
+ }
+ angular.element('.sensor_description_preview').hide();
+ angular.element('.sensor_description_full').show();
+ });
+ }
}
-
- }
})();
(function() {
'use strict';
angular.module('app.components')
- .controller('LayoutController', LayoutController);
-
- LayoutController.$inject = ['$mdSidenav','$mdDialog', '$location', '$state', '$scope', '$transitions', 'auth', 'animation', '$timeout', 'DROPDOWN_OPTIONS_COMMUNITY', 'DROPDOWN_OPTIONS_USER'];
- function LayoutController($mdSidenav, $mdDialog, $location, $state, $scope, $transitions, auth, animation, $timeout, DROPDOWN_OPTIONS_COMMUNITY, DROPDOWN_OPTIONS_USER) {
- var vm = this;
+ .directive('moveFilters', moveFilters);
- vm.navRightLayout = 'space-around center';
+ /**
+ * Moves map filters when scrolling
+ *
+ */
+ moveFilters.$inject = ['$window', '$timeout'];
+ function moveFilters($window, $timeout) {
+ return {
+ link: link
+ };
- $scope.toggleRight = buildToggler('right');
+ function link() {
+ var chartHeight;
+ $timeout(function() {
+ chartHeight = angular.element('.kit_chart').height();
+ }, 1000);
- function buildToggler(componentId) {
- return function() {
- $mdSidenav(componentId).toggle();
- };
+ /*
+ angular.element($window).on('scroll', function() {
+ var windowPosition = document.body.scrollTop;
+ if(chartHeight > windowPosition) {
+ elem.css('bottom', 12 + windowPosition + 'px');
+ }
+ });
+ */
}
+ }
+})();
- // listen for any login event so that the navbar can be updated
- $scope.$on('loggedIn', function(ev, options) {
- // if(options && options.time === 'appLoad') {
- // $scope.$apply(function() {
- // vm.isLoggedin = true;
- // vm.isShown = true;
- // angular.element('.nav_right .wrap-dd-menu').css('display', 'initial');
- // vm.currentUser = auth.getCurrentUser().data;
- // vm.dropdownOptions[0].text = 'Hello, ' + vm.currentUser.username;
- // vm.navRightLayout = 'end center';
- // });
- // } else {
- // vm.isLoggedin = true;
- // vm.isShown = true;
- // angular.element('.nav_right .wrap-dd-menu').css('display', 'initial');
- // vm.currentUser = auth.getCurrentUser().data;
- // vm.dropdownOptions[0].text = 'Hello, ' + vm.currentUser.username;
- // vm.navRightLayout = 'end center';
- // }
-
- vm.isLoggedin = true;
- vm.isShown = true;
- angular.element('.nav_right .wrap-dd-menu').css('display', 'initial');
- vm.currentUser = auth.getCurrentUser().data;
- vm.dropdownOptions[0].text = 'Hi, ' + vm.currentUser.username + '!';
- vm.navRightLayout = 'end center';
- if(!$scope.$$phase) {
- $scope.$digest();
- }
- });
-
- // listen for logout events so that the navbar can be updated
- $scope.$on('loggedOut', function() {
- vm.isLoggedIn = false;
- vm.isShown = true;
- angular.element('navbar .wrap-dd-menu').css('display', 'none');
- vm.navRightLayout = 'space-around center';
- });
-
-
- vm.isShown = true;
- vm.isLoggedin = false;
- vm.logout = logout;
-
- vm.dropdownOptions = DROPDOWN_OPTIONS_USER;
- vm.dropdownSelected = undefined;
+(function() {
+ 'use strict';
- vm.dropdownOptionsCommunity = DROPDOWN_OPTIONS_COMMUNITY;
- vm.dropdownSelectedCommunity = undefined;
+ angular.module('app.components')
+ .factory('layout', layout);
- $scope.$on('removeNav', function() {
- vm.isShown = false;
- });
- $scope.$on('addNav', function() {
- vm.isShown = true;
- });
+ function layout() {
- initialize();
+ var kitHeight;
- //////////////////
+ var service = {
+ setKit: setKit,
+ getKit: getKit
+ };
+ return service;
- function initialize() {
- $timeout(function() {
- var hash = $location.search();
- if(hash.signup) {
- animation.showSignup();
- } else if(hash.login) {
- animation.showLogin();
- } else if(hash.passwordRecovery) {
- animation.showPasswordRecovery();
- }
- }, 1000);
+ function setKit(height) {
+ kitHeight = height;
}
- function logout() {
- auth.logout();
- vm.isLoggedin = false;
+ function getKit() {
+ return kitHeight;
}
}
})();
@@ -3848,1103 +3827,1160 @@ angular.module('app.components')
'use strict';
angular.module('app.components')
- .controller('LandingController', LandingController);
-
- LandingController.$inject = ['$timeout', 'animation', '$mdDialog', '$location', '$anchorScroll'];
+ .directive('horizontalScroll', horizontalScroll);
- function LandingController($timeout, animation, $mdDialog, $location, $anchorScroll) {
- var vm = this;
+ /**
+ * Used to highlight and unhighlight buttons on the kit dashboard when scrolling horizontally
+ *
+ */
+ horizontalScroll.$inject = ['$window', '$timeout'];
+ function horizontalScroll($window, $timeout) {
+ return {
+ link: link,
+ restrict: 'A'
+ };
- vm.showStore = showStore;
- vm.goToHash = goToHash;
+ ///////////////////
- ///////////////////////
- initialize();
+ function link(scope, element) {
- //////////////////
+ element.on('scroll', function() {
+ // horizontal scroll position
+ var position = angular.element(this).scrollLeft();
+ // real width of element
+ var scrollWidth = this.scrollWidth;
+ // visible width of element
+ var width = angular.element(this).width();
- function initialize() {
- $timeout(function() {
- animation.viewLoaded();
- if($location.hash()) {
- $anchorScroll();
+ // if you cannot scroll, unhighlight both
+ if(scrollWidth === width) {
+ angular.element('.button_scroll_left').css('opacity', '0.5');
+ angular.element('.button_scroll_right').css('opacity', '0.5');
}
- }, 500);
- }
-
- function goToHash(hash){
- $location.hash(hash);
- $anchorScroll();
- }
+ // if scroll is in the middle, highlight both
+ if(scrollWidth - width > 2) {
+ angular.element('.button_scroll_left').css('opacity', '1');
+ angular.element('.button_scroll_right').css('opacity', '1');
+ }
+ // if scroll is at the far right, unhighligh right button
+ if(scrollWidth - width - position <= 2) {
+ angular.element('.button_scroll_right').css('opacity', '0.5');
+ return;
+ }
+ // if scroll is at the far left, unhighligh left button
+ if(position === 0) {
+ angular.element('.button_scroll_left').css('opacity', '0.5');
+ return;
+ }
- function showStore() {
- $mdDialog.show({
- hasBackdrop: true,
- controller: 'StoreModalController',
- templateUrl: 'app/components/store/storeModal.html',
- clickOutsideToClose: true
+ //set opacity back to normal otherwise
+ angular.element('.button_scroll_left').css('opacity', '1');
+ angular.element('.button_scroll_right').css('opacity', '1');
});
- }
- }
-})();
-(function(){
- 'use strict';
- angular.module('app.components')
- .directive('kitList',kitList);
+ $timeout(function() {
+ element.trigger('scroll');
+ });
- function kitList(){
- return{
- restrict:'E',
- scope:{
- devices:'=devices',
- actions: '=actions'
- },
- controllerAs:'vm',
- templateUrl:'app/components/kitList/kitList.html'
- };
+ angular.element($window).on('resize', function() {
+ $timeout(function() {
+ element.trigger('scroll');
+ }, 1000);
+ });
+ }
}
})();
(function() {
- 'use strict';
+ 'use strict';
- angular.module('app.components')
- .controller('HomeController', HomeController);
+ angular.module('app.components')
+ .directive('hidePopup', hidePopup);
- function HomeController() {
- }
-})();
-(function (){
- 'use strict';
-
- angular.module('app.components')
- .controller('DownloadModalController', DownloadModalController);
-
- DownloadModalController.$inject = ['thisDevice', 'device', '$mdDialog'];
-
- function DownloadModalController(thisDevice, device, $mdDialog) {
- var vm = this;
-
- vm.device = thisDevice;
- vm.download = download;
- vm.cancel = cancel;
-
- ////////////////////////////
-
- function download(){
- device.mailReadings(vm.device)
- .then(function (){
- $mdDialog.hide();
- }).catch(function(err){
- $mdDialog.cancel(err);
- });
- }
+ /**
+ * Used on kit dashboard to hide popup with full sensor description
+ *
+ */
+
+ hidePopup.$inject = [];
+ function hidePopup() {
+ return {
+ link: link
+ };
- function cancel(){
- $mdDialog.cancel();
- }
- }
+ /////////////
+ function link(scope, elem) {
+ elem.on('mouseleave', function() {
+ angular.element('.sensor_description_preview').show();
+ angular.element('.sensor_description_full').hide();
+ });
+ }
+ }
})();
-(function(){
-'use strict';
+(function() {
+ 'use strict';
-angular.module('app.components')
- .directive('cookiesLaw', cookiesLaw);
+ angular.module('app.components')
+ .directive('disableScroll', disableScroll);
+ disableScroll.$inject = ['$timeout'];
+ function disableScroll($timeout) {
+ return {
+ // link: {
+ // pre: link
+ // },
+ compile: link,
+ restrict: 'A',
+ priority: 100000
+ };
-cookiesLaw.$inject = ['$cookies'];
-function cookiesLaw($cookies) {
- return {
- template:
- '' +
- 'This site uses cookies to offer you a better experience. ' +
- '
Accept or' +
- '
Learn More. ' +
- '
',
- controller: function($scope) {
+ //////////////////////
- var init = function(){
- $scope.isCookieValid();
+ function link(elem) {
+ console.log('i', elem);
+ // var select = elem.find('md-select');
+ // angular.element(select).on('click', function() {
+ elem.on('click', function() {
+ console.log('e');
+ angular.element(document.body).css('overflow', 'hidden');
+ $timeout(function() {
+ angular.element(document.body).css('overflow', 'initial');
+ });
+ });
}
+ }
+})();
- // Helpers to debug
- // You can also use `document.cookie` in the browser dev console.
- //console.log($cookies.getAll());
+(function() {
+ 'use strict';
- $scope.isCookieValid = function() {
- // Use a boolean for the ng-hide, because using a function with ng-hide
- // is considered bad practice. The digest cycle will call it multiple
- // times, in our case around 240 times.
- $scope.isCookieValidBool = ($cookies.get('consent') === 'true')
- }
+ angular.module('app.components')
+ .factory('animation', animation);
- $scope.acceptCookie = function() {
- //console.log('Accepting cookie...');
- var today = new Date();
- var expireDate = new Date(today);
- expireDate.setMonth(today.getMonth() + 6);
+ /**
+ * Used to emit events from rootscope.
+ *
+ * This events are then listened by $scope on controllers and directives that care about that particular event
+ */
- $cookies.put('consent', true, {'expires' : expireDate.toUTCString()} );
+ animation.$inject = ['$rootScope'];
+ function animation($rootScope) {
- // Trigger the check again, after we click
- $scope.isCookieValid();
- };
+ var service = {
+ blur: blur,
+ unblur: unblur,
+ removeNav: removeNav,
+ addNav: addNav,
+ showChartSpinner: showChartSpinner,
+ hideChartSpinner: hideChartSpinner,
+ deviceLoaded: deviceLoaded,
+ showPasswordRecovery: showPasswordRecovery,
+ showLogin: showLogin,
+ showSignup: showSignup,
+ showPasswordReset: showPasswordReset,
+ hideAlert: hideAlert,
+ viewLoading: viewLoading,
+ viewLoaded: viewLoaded,
+ deviceWithoutData: deviceWithoutData,
+ deviceIsPrivate: deviceIsPrivate,
+ goToLocation: goToLocation,
+ mapStateLoading: mapStateLoading,
+ mapStateLoaded: mapStateLoaded
+ };
+ return service;
- init();
+ //////////////
+ function blur() {
+ $rootScope.$broadcast('blur');
+ }
+ function unblur() {
+ $rootScope.$broadcast('unblur');
+ }
+ function removeNav() {
+ $rootScope.$broadcast('removeNav');
+ }
+ function addNav() {
+ $rootScope.$broadcast('addNav');
+ }
+ function showChartSpinner() {
+ $rootScope.$broadcast('showChartSpinner');
+ }
+ function hideChartSpinner() {
+ $rootScope.$broadcast('hideChartSpinner');
+ }
+ function deviceLoaded(data) {
+ $rootScope.$broadcast('deviceLoaded', data);
+ }
+ function showPasswordRecovery() {
+ $rootScope.$broadcast('showPasswordRecovery');
+ }
+ function showLogin() {
+ $rootScope.$broadcast('showLogin');
+ }
+ function showSignup() {
+ $rootScope.$broadcast('showSignup');
+ }
+ function showPasswordReset() {
+ $rootScope.$broadcast('showPasswordReset');
+ }
+ function hideAlert() {
+ $rootScope.$broadcast('hideAlert');
+ }
+ function viewLoading() {
+ $rootScope.$broadcast('viewLoading');
+ }
+ function viewLoaded() {
+ $rootScope.$broadcast('viewLoaded');
+ }
+ function deviceWithoutData(data) {
+ $rootScope.$broadcast('deviceWithoutData', data);
+ }
+ function deviceIsPrivate(data) {
+ $rootScope.$broadcast('deviceIsPrivate', data);
+ }
+ function goToLocation(data) {
+ $rootScope.$broadcast('goToLocation', data);
+ }
+ function mapStateLoading() {
+ $rootScope.$broadcast('mapStateLoading');
+ }
+ function mapStateLoaded() {
+ $rootScope.$broadcast('mapStateLoaded');
+ }
}
- };
-}
-
-
})();
(function() {
'use strict';
- angular.module('app.components')
- .directive('chart', chart);
+ /**
+ * TODO: Improvement These directives can be split up each one in a different file
+ */
- chart.$inject = ['sensor', 'animation', '$timeout', '$window'];
- function chart(sensor, animation, $timeout, $window) {
- var margin, width, height, svg, xScale, yScale0, yScale1, xAxis, yAxisLeft, yAxisRight, dateFormat, areaMain, valueLineMain, areaCompare, valueLineCompare, focusCompare, focusMain, popup, dataMain, colorMain, yAxisScale, unitMain, popupContainer;
+ angular.module('app.components')
+ .directive('moveDown', moveDown)
+ .directive('stick', stick)
+ .directive('blur', blur)
+ .directive('focus', focus)
+ .directive('changeMapHeight', changeMapHeight)
+ .directive('changeContentMargin', changeContentMargin)
+ .directive('focusInput', focusInput);
+
+ /**
+ * It moves down kit section to ease the transition after the kit menu is sticked to the top
+ *
+ */
+ moveDown.$inject = [];
+ function moveDown() {
+
+ function link(scope, element) {
+ scope.$watch('moveDown', function(isTrue) {
+ if(isTrue) {
+ element.addClass('move_down');
+ } else {
+ element.removeClass('move_down');
+ }
+ });
+ }
return {
link: link,
- restrict: 'A',
- scope: {
- chartData: '='
- }
+ scope: false,
+ restrict: 'A'
};
+ }
- function link(scope, elem) {
-
- $timeout(function() {
- createChart(elem[0]);
- }, 0);
-
- var lastData = {};
-
- // on window resize, it re-renders the chart to fit into the new window size
- angular.element($window).on('resize', function() {
- createChart(elem[0]);
- updateChartData(lastData.data, {type: lastData.type, container: elem[0], color: lastData.color, unit: lastData.unit});
- });
+ /**
+ * It sticks kit menu when kit menu touchs navbar on scrolling
+ *
+ */
+ stick.$inject = ['$window', '$timeout'];
+ function stick($window, $timeout) {
+ function link(scope, element) {
+ var elementPosition = element[0].offsetTop;
+ //var elementHeight = element[0].offsetHeight;
+ var navbarHeight = angular.element('.stickNav').height();
- scope.$watch('chartData', function(newData) {
- if(!newData) {
- return;
- }
+ $timeout(function() {
+ elementPosition = element[0].offsetTop;
+ //var elementHeight = element[0].offsetHeight;
+ navbarHeight = angular.element('.stickNav').height();
+ }, 1000);
- if(newData !== undefined) {
- // if there's data for 2 sensors
- if(newData[0] && newData[1]) {
- var sensorDataMain = newData[0].data;
- // we could get some performance from saving the map in the showKit controller on line 218 and putting that logic in here
- var dataMain = sensorDataMain.map(function(dataPoint) {
- return {
- date: dateFormat(dataPoint.time),
- count: dataPoint && dataPoint.count,
- value: dataPoint && dataPoint.value
- };
- });
- // sort data points by date
- dataMain.sort(function(a, b) {
- return a.date - b.date;
- });
- var sensorDataCompare = newData[1].data;
- var dataCompare = sensorDataCompare.map(function(dataPoint) {
- return {
- date: dateFormat(dataPoint.time),
- count: dataPoint && dataPoint.count,
- value: dataPoint && dataPoint.value
- };
- });
+ angular.element($window).on('scroll', function() {
+ var windowPosition = document.body.scrollTop;
- dataCompare.sort(function(a, b) {
- return a.date - b.date;
- });
+ //sticking menu and moving up/down
+ if(windowPosition + navbarHeight >= elementPosition) {
+ element.addClass('stickMenu');
+ scope.$apply(function() {
+ scope.moveDown = true;
+ });
+ } else {
+ element.removeClass('stickMenu');
+ scope.$apply(function() {
+ scope.moveDown = false;
+ });
+ }
+ });
+ }
- var data = [dataMain, dataCompare];
- var colors = [newData[0].color, newData[1].color];
- var units = [newData[0].unit, newData[1].unit];
- // saves everything in case we need to re-render
- lastData = {
- data: data,
- type: 'both',
- color: colors,
- unit: units
- };
- // call function to update the chart with the new data
- updateChartData(data, {type: 'both', container: elem[0], color: colors, unit: units });
- // if only data for the main sensor
- } else if(newData[0]) {
+ return {
+ link: link,
+ scope: false,
+ restrict: 'A'
+ };
+ }
- var sensorData = newData[0].data;
- /*jshint -W004 */
- var data = sensorData.map(function(dataPoint) {
- return {
- date: dateFormat(dataPoint.time),
- count: dataPoint && dataPoint.count,
- value: dataPoint && dataPoint.value
- };
- });
+ /**
+ * Unused directive. Double-check is not being used before removing it
+ *
+ */
- data.sort(function(a, b) {
- return a.date - b.date;
- });
+ function blur() {
- var color = newData[0].color;
- var unit = newData[0].unit;
+ function link(scope, element) {
- lastData = {
- data: data,
- type: 'main',
- color: color,
- unit: unit
- };
+ scope.$on('blur', function() {
+ element.addClass('blur');
+ });
- updateChartData(data, {type: 'main', container: elem[0], color: color, unit: unit });
- }
- animation.hideChartSpinner();
- }
+ scope.$on('unblur', function() {
+ element.removeClass('blur');
});
}
- // creates the container that is re-used across different sensor charts
- function createChart(elem) {
- d3.select(elem).selectAll('*').remove();
+ return {
+ link: link,
+ scope: false,
+ restrict: 'A'
+ };
+ }
- margin = {top: 20, right: 12, bottom: 20, left: 42};
- width = elem.clientWidth - margin.left - margin.right;
- height = elem.clientHeight - margin.top - margin.bottom;
+ /**
+ * Used to remove nav and unable scrolling when searching
+ *
+ */
+ focus.$inject = ['animation'];
+ function focus(animation) {
+ function link(scope, element) {
+ element.on('focusin', function() {
+ animation.removeNav();
+ });
- xScale = d3.time.scale().range([0, width]);
- xScale.tickFormat("%Y-%m-%d %I:%M:%S");
- yScale0 = d3.scale.linear().range([height, 0]);
- yScale1 = d3.scale.linear().range([height, 0]);
- yAxisScale = d3.scale.linear().range([height, 0]);
+ element.on('focusout', function() {
+ animation.addNav();
+ });
- dateFormat = d3.time.format('%Y-%m-%dT%H:%M:%S').parse;//d3.time.format('%Y-%m-%dT%X.%LZ').parse; //'YYYY-MM-DDTHH:mm:ssZ'
+ var searchInput = element.find('input');
+ searchInput.on('blur', function() {
+ //enable scrolling on body when search input is not active
+ angular.element(document.body).css('overflow', 'auto');
+ });
- xAxis = d3.svg.axis()
- .scale(xScale)
- .orient('bottom')
- .ticks(5);
+ searchInput.on('focus', function() {
+ angular.element(document.body).css('overflow', 'hidden');
+ });
+ }
- yAxisLeft = d3.svg.axis()
- .scale(yScale0)
- .orient('left')
- .ticks(5);
+ return {
+ link: link
+ };
+ }
- yAxisRight = d3.svg.axis()
- .scale(yScale1)
- .orient('right')
- .ticks(5);
+ /**
+ * Changes map section based on screen size
+ *
+ */
+ changeMapHeight.$inject = ['$document', 'layout', '$timeout'];
+ function changeMapHeight($document, layout, $timeout) {
+ function link(scope, element) {
- areaMain = d3.svg.area()
- .defined(function(d) {return d.value != null })
- .interpolate('linear')
- .x(function(d) { return xScale(d.date); })
- .y0(height)
- .y1(function(d) { return yScale0(d.count); });
+ var screenHeight = $document[0].body.clientHeight;
+ var navbarHeight = angular.element('.stickNav').height();
- valueLineMain = d3.svg.line()
- .defined(function(d) {return d.value != null })
- .interpolate('linear')
- .x(function(d) { return xScale(d.date); })
- .y(function(d) { return yScale0(d.count); });
+ // var overviewHeight = angular.element('.kit_overview').height();
+ // var menuHeight = angular.element('.kit_menu').height();
+ // var chartHeight = angular.element('.kit_chart').height();
- areaCompare = d3.svg.area()
- .defined(function(d) {return d.value != null })
- .interpolate('linear')
- .x(function(d) { return xScale(d.date); })
- .y0(height)
- .y1(function(d) { return yScale1(d.count); });
+ function resizeMap(){
+ $timeout(function() {
+ var overviewHeight = angular.element('.over_map').height();
- valueLineCompare = d3.svg.line()
- .defined(function(d) {return d.value != null })
- .interpolate('linear')
- .x(function(d) { return xScale(d.date); })
- .y(function(d) { return yScale1(d.count); });
+ var objectsHeight = navbarHeight + overviewHeight;
+ var objectsHeightPercentage = parseInt((objectsHeight * 100) / screenHeight);
+ var mapHeightPercentage = 100 - objectsHeightPercentage;
- svg = d3
- .select(elem)
- .append('svg')
- .attr('width', width + margin.left + margin.right)
- .attr('height', height + margin.top + margin.bottom)
- .append('g')
- .attr('transform', 'translate(' + (margin.left - margin.right) + ',' + margin.top + ')');
- }
- // calls functions depending on type of chart
- function updateChartData(newData, options) {
- if(options.type === 'main') {
- updateChartMain(newData, options);
- } else if(options.type === 'both') {
- updateChartCompare(newData, options);
+ element.css('height', mapHeightPercentage + '%');
+
+ var aboveTheFoldHeight = screenHeight - overviewHeight;
+ angular
+ .element('section[change-content-margin]')
+ .css('margin-top', aboveTheFoldHeight + 'px');
+ });
}
- }
- // function in charge of rendering when there's data for 1 sensor
- function updateChartMain(data, options) {
- xScale.domain(d3.extent(data, function(d) { return d.date; }));
- yScale0.domain([(d3.min(data, function(d) { return d.count; })) * 0.8, (d3.max(data, function(d) { return d.count; })) * 1.2]);
- svg.selectAll('*').remove();
+ resizeMap();
- //Add the area path
- svg.append('path')
- .datum(data)
- .attr('class', 'chart_area')
- .attr('fill', options.color)
- .attr('d', areaMain);
+ scope.element = element;
- // Add the valueline path.
- svg.append('path')
- .attr('class', 'chart_line')
- .attr('stroke', options.color)
- .attr('d', valueLineMain(data));
-
- // Add the X Axis
- svg.append('g')
- .attr('class', 'axis x')
- .attr('transform', 'translate(0,' + height + ')')
- .call(xAxis);
-
- // Add the Y Axis
- svg.append('g')
- .attr('class', 'axis y_left')
- .call(yAxisLeft);
+ scope.$on('resizeMapHeight',function(){
+ resizeMap();
+ });
- // Draw the x Grid lines
- svg.append('g')
- .attr('class', 'grid')
- .attr('transform', 'translate(0,' + height + ')')
- .call(xGrid()
- .tickSize(-height, 0, 0)
- .tickFormat('')
- );
+ }
- // Draw the y Grid lines
- svg.append('g')
- .attr('class', 'grid')
- .call(yGrid()
- .tickSize(-width, 0, 0)
- .tickFormat('')
- );
+ return {
+ link: link,
+ scope: true,
+ restrict: 'A'
+ };
+ }
- focusMain = svg.append('g')
- .attr('class', 'focus')
- .style('display', 'none');
+ /**
+ * Changes margin on kit section based on above-the-fold space left after map section is resize
+ */
- focusMain.append('circle')
- .style('stroke', options.color)
- .attr('r', 4.5);
+ changeContentMargin.$inject = ['layout', '$timeout', '$document'];
+ function changeContentMargin(layout, $timeout, $document) {
+ function link(scope, element) {
+ var screenHeight = $document[0].body.clientHeight;
- var popupWidth = 84;
- var popupHeight = 46;
+ var overviewHeight = angular.element('.over_map').height();
- popup = svg.append('g')
- .attr('class', 'focus')
- .style('display', 'none');
+ var aboveTheFoldHeight = screenHeight - overviewHeight;
+ element.css('margin-top', aboveTheFoldHeight + 'px');
+ }
- popupContainer = popup.append('rect')
- .attr('width', popupWidth)
- .attr('height', popupHeight)
- .attr('transform', function() {
- var result = 'translate(-42, 5)';
+ return {
+ link: link
+ };
+ }
- return result;
- })
- .style('stroke', 'grey')
- .style('stroke-width', '0.5')
- .style('fill', 'white');
+ /**
+ * Fixes autofocus for inputs that are inside modals
+ *
+ */
+ focusInput.$inject = ['$timeout'];
+ function focusInput($timeout) {
+ function link(scope, elem) {
+ $timeout(function() {
+ elem.focus();
+ });
+ }
+ return {
+ link: link
+ };
+ }
+})();
- var text = popup.append('text')
- .attr('class', '');
+(function() {
+ 'use strict';
- var textMain = text.append('tspan')
- .attr('class', 'popup_main')
- .attr('text-anchor', 'start')
- .attr('x', -popupWidth / 2)
- .attr('dx', 8)
- .attr('y', popupHeight / 2)
- .attr('dy', 3);
+ angular.module('app.components')
+ .directive('activeButton', activeButton);
- textMain.append('tspan')
- .attr('class', 'popup_value');
+ /**
+ * Used to highlight and unhighlight buttons on kit menu
+ *
+ * It attaches click handlers dynamically
+ */
- textMain.append('tspan')
- .attr('class', 'popup_unit')
- .attr('dx', 5);
+ activeButton.$inject = ['$timeout', '$window'];
+ function activeButton($timeout, $window) {
+ return {
+ link: link,
+ restrict: 'A'
- text.append('tspan')
- .attr('class', 'popup_date')
- .attr('x', -popupWidth / 2)
- .attr('dx', 8)
- .attr('y', popupHeight - 2)
- .attr('dy', 0)
- .attr( 'text-anchor', 'start' );
+ };
- svg.append('rect')
- .attr('class', 'overlay')
- .attr('width', width)
- .attr('height', height)
- .on('mouseover', function() {
- popup.style('display', null);
- focusMain.style('display', null);
- })
- .on('mouseout', function() {
- popup.style('display', 'none');
- focusMain.style('display', 'none');
- })
- .on('mousemove', mousemove);
+ ////////////////////////////
+ function link(scope, element) {
+ var childrens = element.children();
+ var container;
+ $timeout(function() {
+ var navbar = angular.element('.stickNav');
+ var kitMenu = angular.element('.kit_menu');
+ var kitOverview = angular.element('.kit_overview');
+ var kitDashboard = angular.element('.kit_chart');
+ var kitDetails = angular.element('.kit_details');
+ var kitOwner = angular.element('.kit_owner');
+ var kitComments = angular.element('.kit_comments');
- function mousemove() {
- var bisectDate = d3.bisector(function(d) { return d.date; }).left;
+ container = {
+ navbar: {
+ height: navbar.height()
+ },
+ kitMenu: {
+ height: kitMenu.height()
+ },
+ kitOverview: {
+ height: kitOverview.height(),
+ offset: kitOverview.offset().top,
+ buttonOrder: 0
+ },
+ kitDashboard: {
+ height: kitDashboard.height(),
+ offset: kitDashboard.offset().top,
+ buttonOrder: 40
+ },
+ kitDetails: {
+ height: kitDetails.height(),
+ offset: kitDetails.offset() ? kitDetails.offset().top : 0,
+ buttonOrder: 1
+ },
+ kitOwner: {
+ height: kitOwner.height(),
+ offset: kitOwner.offset() ? kitOwner.offset().top : 0,
+ buttonOrder: 2
+ },
+ kitComments: {
+ height: kitComments.height(),
+ offset: kitComments.offset() ? kitComments.offset().top : 0,
+ buttonOrder: 3
+ }
+ };
+ }, 1000);
- var x0 = xScale.invert(d3.mouse(this)[0]);
- var i = bisectDate(data, x0, 1);
- var d0 = data[i - 1];
- var d1 = data[i];
- var d = d1 && (x0 - d0.date > d1.date - x0) ? d1 : d0;
+ function scrollTo(offset) {
+ if(!container) {
+ return;
+ }
+ angular.element($window).scrollTop(offset - container.navbar.height - container.kitMenu.height);
+ }
- focusMain.attr('transform', 'translate(' + xScale(d.date) + ', ' + yScale0(d.count) + ')');
- var popupText = popup.select('text');
- var textMain = popupText.select('.popup_main');
- var valueMain = textMain.select('.popup_value').text(parseValue(d.value));
- var unitMain = textMain.select('.popup_unit').text(options.unit);
- var date = popupText.select('.popup_date').text(parseTime(d.date));
+ function getButton(buttonOrder) {
+ return childrens[buttonOrder];
+ }
- var textContainers = [
- textMain,
- date
- ];
+ function unHighlightButtons() {
+ //remove border, fill and stroke of every icon
+ var activeButton = angular.element('.md-button.button_active');
+ if(activeButton.length) {
+ activeButton.removeClass('button_active');
- var popupWidth = resizePopup(popupContainer, textContainers);
+ var strokeContainer = activeButton.find('.stroke_container');
+ strokeContainer.css('stroke', 'none');
+ strokeContainer.css('stroke-width', '1');
- if(xScale(d.date) + 80 + popupWidth > options.container.clientWidth) {
- popup.attr('transform', 'translate(' + (xScale(d.date) - 120) + ', ' + (d3.mouse(this)[1] - 20) + ')');
- } else {
- popup.attr('transform', 'translate(' + (xScale(d.date) + 80) + ', ' + (d3.mouse(this)[1] - 20) + ')');
+ var fillContainer = strokeContainer.find('.fill_container');
+ fillContainer.css('fill', '#FF8600');
}
}
- }
-
- // function in charge of rendering when there's data for 2 sensors
- function updateChartCompare(data, options) {
- xScale.domain(d3.extent(data[0], function(d) { return d.date; }));
- yScale0.domain([(d3.min(data[0], function(d) { return d.count; })) * 0.8, (d3.max(data[0], function(d) { return d.count; })) * 1.2]);
- yScale1.domain([(d3.min(data[1], function(d) { return d.count; })) * 0.8, (d3.max(data[1], function(d) { return d.count; })) * 1.2]);
- svg.selectAll('*').remove();
+ function highlightButton(button) {
+ var clickedButton = angular.element(button);
+ //add border, fill and stroke to every icon
+ clickedButton.addClass('button_active');
- //Add both area paths
- svg.append('path')
- .datum(data[0])
- .attr('class', 'chart_area')
- .attr('fill', options.color[0])
- .attr('d', areaMain);
+ var strokeContainer = clickedButton.find('.stroke_container');
+ strokeContainer.css('stroke', 'white');
+ strokeContainer.css('stroke-width', '0.01px');
- svg.append('path')
- .datum(data[1])
- .attr('class', 'chart_area')
- .attr('fill', options.color[1])
- .attr('d', areaCompare);
+ var fillContainer = strokeContainer.find('.fill_container');
+ fillContainer.css('fill', 'white');
+ }
- // Add both valueline paths.
- svg.append('path')
- .attr('class', 'chart_line')
- .attr('stroke', options.color[0])
- .attr('d', valueLineMain(data[0]));
-
- svg.append('path')
- .attr('class', 'chart_line')
- .attr('stroke', options.color[1])
- .attr('d', valueLineCompare(data[1]));
+ //attach event handlers for clicks for every button and scroll to a section when clicked
+ _.each(childrens, function(button) {
+ angular.element(button).on('click', function() {
+ var buttonOrder = angular.element(this).index();
+ for(var elem in container) {
+ if(container[elem].buttonOrder === buttonOrder) {
+ var offset = container[elem].offset;
+ scrollTo(offset);
+ angular.element($window).trigger('scroll');
+ }
+ }
+ });
+ });
- // Add the X Axis
- svg.append('g')
- .attr('class', 'axis x')
- .attr('transform', 'translate(0,' + height + ')')
- .call(xAxis);
+ var currentSection;
- // Add both Y Axis
- svg.append('g')
- .attr('class', 'axis y_left')
- .call(yAxisLeft);
+ //on scroll, check if window is on a section
+ angular.element($window).on('scroll', function() {
+ if(!container){ return; }
- svg.append('g')
- .attr('class', 'axis y_right')
- .attr('transform', 'translate(' + width + ' ,0)')
- .call(yAxisRight);
+ var windowPosition = document.body.scrollTop;
+ var appPosition = windowPosition + container.navbar.height + container.kitMenu.height;
+ var button;
+ if(currentSection !== 'none' && appPosition <= container.kitOverview.offset) {
+ button = getButton(container.kitOverview.buttonOrder);
+ unHighlightButtons();
+ currentSection = 'none';
+ } else if(currentSection !== 'overview' && appPosition >= container.kitOverview.offset && appPosition <= container.kitOverview.offset + container.kitOverview.height) {
+ button = getButton(container.kitOverview.buttonOrder);
+ unHighlightButtons();
+ highlightButton(button);
+ currentSection = 'overview';
+ } else if(currentSection !== 'details' && appPosition >= container.kitDetails.offset && appPosition <= container.kitDetails.offset + container.kitDetails.height) {
+ button = getButton(container.kitDetails.buttonOrder);
+ unHighlightButtons();
+ highlightButton(button);
+ currentSection = 'details';
+ } else if(currentSection !== 'owner' && appPosition >= container.kitOwner.offset && appPosition <= container.kitOwner.offset + container.kitOwner.height) {
+ button = getButton(container.kitOwner.buttonOrder);
+ unHighlightButtons();
+ highlightButton(button);
+ currentSection = 'owner';
+ } else if(currentSection !== 'comments' && appPosition >= container.kitComments.offset && appPosition <= container.kitComments.offset + container.kitOwner.height) {
+ button = getButton(container.kitComments.buttonOrder);
+ unHighlightButtons();
+ highlightButton(button);
+ currentSection = 'comments';
+ }
+ });
+ }
+ }
+})();
- // Draw the x Grid lines
- svg.append('g')
- .attr('class', 'grid')
- .attr('transform', 'translate(0,' + height + ')')
- .call(xGrid()
- .tickSize(-height, 0, 0)
- .tickFormat('')
- );
+(function() {
+ 'use strict';
- // Draw the y Grid lines
- svg.append('g')
- .attr('class', 'grid')
- .call(yGrid()
- .tickSize(-width, 0, 0)
- .tickFormat('')
- );
+ angular.module('app.components')
+ .controller('UserProfileController', UserProfileController);
- focusCompare = svg.append('g')
- .attr('class', 'focus')
- .style('display', 'none');
+ UserProfileController.$inject = ['$scope', '$stateParams', '$location',
+ 'user', 'auth', 'userUtils', '$timeout', 'animation',
+ 'NonAuthUser', '$q', 'PreviewDevice'];
+ function UserProfileController($scope, $stateParams, $location,
+ user, auth, userUtils, $timeout, animation,
+ NonAuthUser, $q, PreviewDevice) {
- focusMain = svg.append('g')
- .attr('class', 'focus')
- .style('display', 'none');
+ var vm = this;
+ var userID = parseInt($stateParams.id);
- focusCompare.append('circle')
- .style('stroke', options.color[1])
- .attr('r', 4.5);
+ vm.status = undefined;
+ vm.user = {};
+ vm.devices = [];
+ vm.filteredDevices = [];
+ vm.filterDevices = filterDevices;
- focusMain.append('circle')
- .style('stroke', options.color[0])
- .attr('r', 4.5);
+ $scope.$on('loggedIn', function() {
+ var authUser = auth.getCurrentUser().data;
+ if( userUtils.isAuthUser(userID, authUser) ) {
+ $location.path('/profile');
+ }
+ });
- var popupWidth = 84;
- var popupHeight = 75;
+ initialize();
- popup = svg.append('g')
- .attr('class', 'focus')
- .style('display', 'none');
+ //////////////////
- popupContainer = popup.append('rect')
- .attr('width', popupWidth)
- .attr('height', popupHeight)
- .style('min-width', '40px')
- .attr('transform', function() {
- var result = 'translate(-42, 5)';
+ function initialize() {
- return result;
- })
- .style('stroke', 'grey')
- .style('stroke-width', '0.5')
- .style('fill', 'white');
+ user.getUser(userID)
+ .then(function(user) {
+ vm.user = new NonAuthUser(user);
- popup.append('rect')
- .attr('width', 8)
- .attr('height', 2)
- .attr('transform', function() {
- return 'translate(' + (-popupWidth / 2 + 4).toString() + ', 20)';
- })
- .style('fill', options.color[0]);
+ if(!vm.user.devices.length) {
+ return [];
+ }
- popup.append('rect')
- .attr('width', 8)
- .attr('height', 2)
- .attr('transform', function() {
- return 'translate(' + (-popupWidth / 2 + 4).toString() + ', 45)';
- })
- .style('fill', options.color[1]);
+ $q.all(vm.devices = vm.user.devices.map(function(data){
+ return new PreviewDevice(data);
+ }))
- var text = popup.append('text')
- .attr('class', '');
+ }).then(function(error) {
+ if(error && error.status === 404) {
+ $location.url('/404');
+ }
+ });
- var textMain = text.append('tspan')
- .attr('class', 'popup_main')
- .attr('x', -popupHeight / 2 + 7) //position of text
- .attr('dx', 8) //margin given to the element, will be applied to both sides thanks to resizePopup function
- .attr('y', popupHeight / 3)
- .attr('dy', 3);
+ $timeout(function() {
+ setSidebarMinHeight();
+ animation.viewLoaded();
+ }, 500);
+ }
- textMain.append('tspan')
- .attr('class', 'popup_value')
- .attr( 'text-anchor', 'start' );
+ function filterDevices(status) {
+ if(status === 'all') {
+ status = undefined;
+ }
+ vm.status = status;
+ }
- textMain.append('tspan')
- .attr('class', 'popup_unit')
- .attr('dx', 5);
+ function setSidebarMinHeight() {
+ var height = document.body.clientHeight / 4 * 3;
+ angular.element('.profile_content').css('min-height', height + 'px');
+ }
+ }
+})();
- var textCompare = text.append('tspan')
- .attr('class', 'popup_compare')
- .attr('x', -popupHeight / 2 + 7) //position of text
- .attr('dx', 8) //margin given to the element, will be applied to both sides thanks to resizePopup function
- .attr('y', popupHeight / 1.5)
- .attr('dy', 3);
+(function() {
+ 'use strict';
- textCompare.append('tspan')
- .attr('class', 'popup_value')
- .attr( 'text-anchor', 'start' );
+ angular.module('app.components')
+ .controller('UploadController', UploadController);
- textCompare.append('tspan')
- .attr('class', 'popup_unit')
- .attr('dx', 5);
+ UploadController.$inject = ['kit', '$state', '$stateParams', 'animation'];
+ function UploadController(kit, $state, $stateParams, animation) {
+ var vm = this;
- text.append('tspan')
- .attr('class', 'popup_date')
- .attr('x', (- popupWidth / 2))
- .attr('dx', 8)
- .attr('y', popupHeight - 2)
- .attr('dy', 0)
- .attr( 'text-anchor', 'start' );
+ vm.kit = kit;
- svg.append('rect')
- .attr('class', 'overlay')
- .attr('width', width)
- .attr('height', height)
- .on('mouseover', function() {
- focusCompare.style('display', null);
- focusMain.style('display', null);
- popup.style('display', null);
- })
- .on('mouseout', function() {
- focusCompare.style('display', 'none');
- focusMain.style('display', 'none');
- popup.style('display', 'none');
- })
- .on('mousemove', mousemove);
+ vm.backToProfile = backToProfile;
- function mousemove() {
- var bisectDate = d3.bisector(function(d) { return d.date; }).left;
+ initialize();
- var x0 = xScale.invert(d3.mouse(this)[0]);
- var i = bisectDate(data[1], x0, 1);
- var d0 = data[1][i - 1];
- var d1 = data[1][i];
- var d = x0 - d0.date > d1.date - x0 ? d1 : d0;
- focusCompare.attr('transform', 'translate(' + xScale(d.date) + ', ' + yScale1(d.count) + ')');
+ /////////////////
+ function initialize() {
+ animation.viewLoaded();
+ }
- var dMain0 = data[0][i - 1];
- var dMain1 = data[0][i];
- var dMain = x0 - dMain0.date > dMain1.date - x0 ? dMain1 : dMain0;
- focusMain.attr('transform', 'translate(' + xScale(dMain.date) + ', ' + yScale0(dMain.count) + ')');
-
- var popupText = popup.select('text');
- var textMain = popupText.select('.popup_main');
- textMain.select('.popup_value').text(parseValue(dMain.value));
- textMain.select('.popup_unit').text(options.unit[0]);
- var textCompare = popupText.select('.popup_compare');
- textCompare.select('.popup_value').text(parseValue(d.value));
- textCompare.select('.popup_unit').text(options.unit[1]);
- var date = popupText.select('.popup_date').text(parseTime(d.date));
+ function backToProfile() {
+ $state.transitionTo('layout.myProfile.kits', $stateParams,
+ { reload: false,
+ inherit: false,
+ notify: true
+ });
+ }
+ }
+})();
- var textContainers = [
- textMain,
- textCompare,
- date
- ];
+(function(){
+'use strict';
- var popupWidth = resizePopup(popupContainer, textContainers);
- if(xScale(d.date) + 80 + popupWidth > options.container.clientWidth) {
- popup.attr('transform', 'translate(' + (xScale(d.date) - 120) + ', ' + (d3.mouse(this)[1] - 20) + ')');
- } else {
- popup.attr('transform', 'translate(' + (xScale(d.date) + 80) + ', ' + (d3.mouse(this)[1] - 20) + ')');
- }
- }
- }
- function xGrid() {
- return d3.svg.axis()
- .scale(xScale)
- .orient('bottom')
- .ticks(5);
- }
+function parseDataForPost(csvArray) {
+ /*
+ EXPECTED PAYLOAD
+ {
+ "data": [{
+ "recorded_at": "2016-06-08 10:30:00",
+ "sensors": [{
+ "id": 22,
+ "value": 21
+ }]
+ }]
+ }
+ */
+ const ids = csvArray[3]; // save ids from the 4th header
+ csvArray.splice(0,4); // remove useless headers
+ return {
+ data: csvArray.map((data) => {
+ return {
+ recorded_at: data.shift(), // get the timestamp from the first column
+ sensors: data.map((value, index) => {
+ return {
+ id: ids[index+1], // get ID of sensor from headers
+ value: value
+ };
+ })
+ .filter((sensor) => sensor.value && sensor.id) // remove empty value or id
+ };
+ })
+ };
+}
- function yGrid() {
- return d3.svg.axis()
- .scale(yScale0)
- .orient('left')
- .ticks(5);
- }
- function parseValue(value) {
- if(value === null) {
- return 'No data on the current timespan';
- } else if(value.toString().indexOf('.') !== -1) {
- var result = value.toString().split('.');
- return result[0] + '.' + result[1].slice(0, 2);
- } else if(value > 99.99) {
- return value.toString();
- } else {
- return value.toString().slice(0, 2);
- }
- }
- function parseTime(time) {
- return moment(time).format('h:mm a ddd Do MMM YYYY');
- }
+controller.$inject = ['device', 'Papa', '$mdDialog', '$q'];
+function controller(device, Papa, $mdDialog, $q) {
+ var vm = this;
+ vm.loadingStatus = false;
+ vm.loadingProgress = 0;
+ vm.loadingType = 'indeterminate';
+ vm.csvFiles = [];
+ vm.$onInit = function() {
+ vm.kitLastUpdate = Math.floor(new Date(vm.kit.time).getTime() / 1000);
+ }
+ vm.onSelect = function() {
+ vm.loadingStatus = true;
+ vm.loadingType = 'indeterminate';
+ }
+ vm.change = function(files, invalidFiles) {
+ let count = 0;
+ vm.invalidFiles = invalidFiles;
+ if (!files) { return; }
+ vm.loadingStatus = true;
+ vm.loadingType = 'determinate';
+ vm.loadingProgress = 0;
+ $q.all(
+ files
+ .filter((file) => vm._checkDuplicate(file))
+ .map((file, index, filteredFiles) => {
+ vm.csvFiles.push(file);
+ return vm._analyzeData(file)
+ .then((result) => {
+ if (result.errors && result.errors.length > 0) {
+ file.parseErrors = result.errors;
+ }
+ const lastTimestamp = Math.floor((new Date(result.data[result.data.length - 1][0])).getTime() / 1000);
+ const isNew = vm.kitLastUpdate < lastTimestamp;
+ file.checked = isNew;
+ file.progress = null;
+ file.isNew = isNew;
+ })
+ .then(() => {
+ count += 1;
+ vm.loadingProgress = (count)/filteredFiles.length * 100;
- function resizePopup(popupContainer, textContainers) {
- if(!textContainers.length) {
- return;
- }
+ });
+ })
+ ).then(() => {
+ vm.loadingStatus = false;
+ }).catch(() => {
+ vm.loadingStatus = false;
+ });
+ }
- var widestElem = textContainers.reduce(function(widestElemSoFar, textContainer) {
- var currentTextContainerWidth = getContainerWidth(textContainer);
- var prevTextContainerWidth = getContainerWidth(widestElemSoFar);
- return prevTextContainerWidth >= currentTextContainerWidth ? widestElemSoFar : textContainer;
- }, textContainers[0]);
+ vm.haveSelectedFiles = function() {
+ return vm.csvFiles && vm.csvFiles.some((file) => file.checked);
+ };
- var margins = widestElem.attr('dx') * 2;
+ vm.haveSelectedNoFiles = function() {
+ return vm.csvFiles && !vm.csvFiles.some((file) => file.checked);
+ };
- popupContainer
- .attr('width', getContainerWidth(widestElem) + margins);
+ vm.haveSelectedAllFiles = function() {
+ return vm.csvFiles && vm.csvFiles.every((file) => file.checked);
+ };
- function getContainerWidth(container) {
- var node = container.node();
- var width;
- if(node.getComputedTextLength) {
- width = node.getComputedTextLength();
- } else if(node.getBoundingClientRect) {
- width = node.getBoundingClientRect().width;
- } else {
- width = node.getBBox().width;
- }
- return width;
- }
- return getContainerWidth(widestElem) + margins;
- }
+ vm.doAction = function() {
+ switch (vm.action) {
+ case 'selectAll':
+ vm.selectAll(true);
+ break;
+ case 'deselectAll':
+ vm.selectAll(false);
+ break;
+ case 'upload':
+ vm.uploadData();
+ break;
+ case 'remove':
+ vm.csvFiles = vm.csvFiles.filter((file) => !file.checked);
+ break;
}
+ vm.action = null;
+ };
-})();
+ vm.selectAll = function(value) {
+ vm.csvFiles.forEach((file) => { file.checked = value });
+ };
-(function(){
- 'use strict';
+ vm.removeFile = function(index) {
+ vm.csvFiles.splice(index, 1);
+ };
+ vm._analyzeData = function(file) {
+ file.progress = true;
+ return Papa.parse(file, {
+ delimiter: ',',
+ dynamicTyping: true,
+ worker: false,
+ skipEmptyLines: true
+ }).catch((err) => {
+ file.progress = null;
+ console('catch',err)
+ });
+ };
- angular.module('app.components')
- .directive('apiKey', apiKey);
+ vm._checkDuplicate = function(file) {
+ if (vm.csvFiles.some((csvFile) => file.name === csvFile.name)) {
+ file.$errorMessages = {};
+ file.$errorMessages.duplicate = true;
+ vm.invalidFiles.push(file);
+ return false;
+ } else {
+ return true;
+ }
+ };
- function apiKey(){
- return {
- scope: {
- apiKey: '=apiKey'
- },
- restrict: 'A',
- controller: 'ApiKeyController',
- controllerAs: 'vm',
- templateUrl: 'app/components/apiKey/apiKey.html'
- };
+ vm.showErrorModal = function(csvFile) {
+ $mdDialog.show({
+ hasBackdrop: true,
+ controller: ['$mdDialog',function($mdDialog) {
+ this.parseErrors = csvFile.parseErrors;
+ this.backEndErrors = csvFile.backEndErrors;
+ this.cancel = function() { $mdDialog.hide(); };
+ }],
+ controllerAs: 'csvFile',
+ templateUrl: 'app/components/upload/errorModal.html',
+ clickOutsideToClose: true
+ });
}
-})();
-(function(){
- 'use strict';
- angular.module('app.components')
- .controller('ApiKeyController', ApiKeyController);
+ vm.uploadData = function() {
+ vm.loadingStatus = true;
+ vm.loadingType = 'indeterminate';
+ vm.loadingProgress = 0;
+ let count = 0;
- ApiKeyController.$inject = ['alert'];
- function ApiKeyController(alert){
- var vm = this;
-
- vm.copied = copied;
- vm.copyFail = copyFail;
-
- ///////////////
-
- function copied(){
- alert.success('API key copied to your clipboard.');
- }
+ $q.all(
+ vm.csvFiles
+ .filter((file) => file.checked && !file.success)
+ .map((file, index, filteredFiles) => {
+ file.progress = true;
+ return vm._analyzeData(file)
+ .then((result) => parseDataForPost(result.data)) // TODO: Improvement remove
+ // TODO: Improvement with workers
+ .then((payload) => device.postReadings(vm.kit, payload))
+ .then(() => {
+ if (vm.loadingType === 'indeterminate') { vm.loadingType = 'determinate'; };
+ file.success = true;
+ file.progress = null;
+ count += 1;
+ vm.loadingProgress = (count)/filteredFiles.length * 100;
+ })
+ .catch((errors) => {
+ console.log(errors);
+ file.detailShowed = true;
+ file.backEndErrors = errors;
+ file.progress = null;
+ });
+ })
+ ).then(() => {
+ vm.loadingStatus = false;
+ })
+ .catch(() => {
+ vm.loadingStatus = false;
+ });
+ }
+};
- function copyFail(err){
- console.log('Copy error: ', err);
- alert.error('Oops! An error occurred copying the api key.');
- }
- }
+angular.module('app.components')
+ .component('scCsvUpload', {
+ templateUrl: 'app/components/upload/csvUpload.html',
+ controller: controller,
+ bindings: {
+ kit: '<'
+ },
+ controllerAs: 'vm'
+ });
})();
(function() {
'use strict';
angular.module('app.components')
- .factory('alert', alert);
+ .controller('tagsController', tagsController);
- alert.$inject = ['$mdToast'];
- function alert($mdToast) {
- var service = {
- success: success,
- error: error,
- info: {
- noData: {
- visitor: infoNoDataVisitor,
- owner: infoNoDataOwner,
- private: infoDataPrivate,
- },
- longTime: infoLongTime,
- // TODO: Refactor, check why this was removed
- // inValid: infoDataInvalid,
- generic: info
- }
- };
+ tagsController.$inject = ['tag', '$scope', 'device', '$state', '$q',
+ 'PreviewDevice', 'animation'
+ ];
- return service;
+ function tagsController(tag, $scope, device, $state, $q, PreviewDevice,
+ animation) {
- ///////////////////
+ var vm = this;
- function success(message) {
- toast('success', message);
- }
+ vm.selectedTags = tag.getSelectedTags();
+ vm.markers = [];
+ vm.kits = [];
+ vm.percActive = 0;
- function error(message) {
- toast('error', message);
- }
+ initialize();
- function infoNoDataVisitor() {
- info('Woah! We couldn\'t locate this kit on the map because it hasn\'t published any data. Leave a ' +
- 'comment to let its owner know.',
- 10000,
- {
- button: 'Leave comment',
- href: 'https://forum.smartcitizen.me/'
- });
- }
+ /////////////////////////////////////////////////////////
- function infoNoDataOwner() {
- info('Woah! We couldn\'t locate this kit on the map because it hasn\'t published any data.',
- 10000);
- }
+ function initialize() {
+ if(vm.selectedTags.length === 0){
+ $state.transitionTo('layout.home.kit');
+ }
- function infoDataPrivate() {
- info('Device not found, or it has been set to private. Leave a ' +
- 'comment to let its owner know you\'re interested.',
- 10000,
- {
- button: 'Leave comment',
- href: 'https://forum.smartcitizen.me/'
+ if (device.getWorldMarkers()) {
+ // If the user has already loaded a prev page and has markers in mem or localstorage
+ updateSelectedTags();
+ } else {
+ // If the user is new we wait the map to load the markers
+ $scope.$on('mapStateLoaded', function(event, data) {
+ updateSelectedTags();
});
+ }
+
}
- // TODO: Refactor, check why this was removed
- // function infoDataInvalid() {
- // info('Device not found, or it has been set to private.',
- // 10000);
- // }
+ function updateSelectedTags(){
- function infoLongTime() {
- info('😅 It looks like this kit hasn\'t posted any data in a long ' +
- 'time. Why not leave a comment to let its owner know?',
- 10000,
- {
- button: 'Leave comment',
- href: 'https://forum.smartcitizen.me/'
- });
- }
+ vm.markers = tag.filterMarkersByTag(device.getWorldMarkers());
- function info(message, delay, options) {
- if(options && options.button) {
- toast('infoButton', message, options, undefined, delay);
+ var onlineMarkers = _.filter(vm.markers, isOnline);
+ if (vm.markers.length === 0) {
+ vm.percActive = 0;
} else {
- toast('info', message, options, undefined, delay);
+ vm.percActive = Math.floor(onlineMarkers.length / vm.markers.length *
+ 100);
}
- }
- function toast(type, message, options, position, delay) {
- position = position === undefined ? 'top': position;
- delay = delay === undefined ? 5000 : delay;
+ animation.viewLoaded();
- $mdToast.show({
- controller: 'AlertController',
- controllerAs: 'vm',
- templateUrl: 'app/components/alert/alert' + type + '.html',
- hideDelay: delay,
- position: position,
- locals: {
- message: message,
- button: options && options.button,
- href: options && options.href
- }
- });
+ getTaggedDevices()
+ .then(function(res){
+ vm.kits = res;
+ });
}
- }
-})();
-(function() {
- 'use strict';
- angular.module('app.components')
- .controller('AlertController', AlertController);
+ function isOnline(marker) {
+ return _.includes(marker.myData.labels, 'online');
+ }
- AlertController.$inject = ['$scope', '$mdToast', 'message', 'button', 'href'];
- function AlertController($scope, $mdToast, message, button, href) {
- var vm = this;
+ function descLastUpdate(o) {
+ return -new Date(o.last_reading_at).getTime();
+ }
- vm.close = close;
- vm.message = message;
- vm.button = button;
- vm.href = href;
+ function getTaggedDevices() {
- // hideAlert will be triggered on state change
- $scope.$on('hideAlert', function() {
- close();
- });
+ var deviceProm = _.map(vm.markers, getMarkerDevice);
- ///////////////////
+ return $q.all(deviceProm)
+ .then(function(devices) {
+ return _.map(_.sortBy(devices, descLastUpdate), toPreviewDevice); // This sort is temp
+ });
+ }
- function close() {
- $mdToast.hide();
- }
+ function toPreviewDevice(dev) {
+ return new PreviewDevice(dev);
+ }
+
+ function getMarkerDevice(marker) {
+ return device.getDevice(marker.myData.id);
}
+ }
+
})();
-(function() {
+(function(){
'use strict';
-
angular.module('app.components')
- .factory('userUtils', userUtils);
-
- function userUtils() {
- var service = {
- isAdmin: isAdmin,
- isAuthUser: isAuthUser
- };
- return service;
+ .directive('tag',tag);
- ///////////
+ function tag(){
+ return{
+ restrict: 'E',
+ scope:{
+ tagName: '=',
+ openTag: '&'
+ },
+ controller:function($scope, $state){
+ $scope.openTag = function(){
+ $state.go('layout.home.tags', {tags:[$scope.tagName]});
+ };
+ },
+ template:'{{tagName}}',
+ link: function(scope, element, attrs){
+ element.addClass('tag');
- function isAdmin(userData) {
- return userData.role === 'admin';
- }
- function isAuthUser(userID, authUserData) {
- return userID === authUserData.id;
+ if(typeof(attrs.clickable) !== 'undefined'){
+ element.bind('click', scope.openTag);
+ }
}
- }
+ };
+ }
})();
(function() {
'use strict';
angular.module('app.components')
- .factory('timeUtils', timeUtils);
+ .controller('StoreModalController', StoreModalController);
- function timeUtils() {
- var service = {
- getSecondsFromDate: getSecondsFromDate,
- getMillisFromDate: getMillisFromDate,
- getCurrentRange: getCurrentRange,
- getToday: getToday,
- getHourBefore: getHourBefore,
- getSevenDaysAgo: getSevenDaysAgo,
- getDateIn: getDateIn,
- convertTime: convertTime,
- formatDate: formatDate,
- isSameDay: isSameDay,
- isWithin15min: isWithin15min,
- isWithin1Month: isWithin1Month,
- isWithin: isWithin,
- isDiffMoreThan15min: isDiffMoreThan15min,
- parseDate: parseDate
- };
- return service;
+ StoreModalController.$inject = ['$scope', '$mdDialog'];
+ function StoreModalController($scope, $mdDialog) {
- ////////////
+ $scope.cancel = function() {
+ $mdDialog.hide();
+ };
+ }
+})();
- function getDateIn(timeMS, format) {
- if(!format) {
- return timeMS;
- }
+(function() {
+ 'use strict';
- var result;
- if(format === 'ms') {
- result = timeMS;
- } else if(format === 's') {
- result = timeMS / 1000;
- } else if(format === 'm') {
- result = timeMS / 1000 / 60;
- } else if(format === 'h') {
- result = timeMS / 1000 / 60 / 60;
- } else if(format === 'd') {
- result = timeMS / 1000 / 60 / 60 / 24;
- }
- return result;
- }
+ angular.module('app.components')
+ .directive('store', store);
- function convertTime(time) {
- return moment(time).toISOString();
+ function store() {
+ return {
+ scope: {
+ isLoggedin: '=logged'
+ },
+ restrict: 'A',
+ controller: 'StoreController',
+ controllerAs: 'vm',
+ templateUrl: 'app/components/store/store.html'
+ };
}
+})();
- function formatDate(time) {
- return moment(time).format('YYYY-MM-DDTHH:mm:ss');
- }
+(function() {
+ 'use strict';
- function getSecondsFromDate(date) {
- return (new Date(date)).getTime();
- }
+ angular.module('app.components')
+ .controller('StoreController', StoreController);
- function getMillisFromDate(date) {
- return (new Date(date)).getTime();
- }
+ StoreController.$inject = ['$scope', '$mdDialog'];
+ function StoreController($scope, $mdDialog) {
- function getCurrentRange(fromDate, toDate) {
- return moment(toDate).diff(moment(fromDate), 'days');
- }
+ $scope.showStore = showStore;
- function getToday() {
- return (new Date()).getTime();
- }
+ $scope.$on('showStore', function() {
+ showStore();
+ });
+
+ ////////////////
- function getSevenDaysAgo() {
- return getSecondsFromDate( getToday() - (7 * 24 * 60 * 60 * 1000) );
+ function showStore() {
+ $mdDialog.show({
+ hasBackdrop: true,
+ controller: 'StoreModalController',
+ templateUrl: 'app/components/store/storeModal.html',
+ clickOutsideToClose: true
+ });
}
- function getHourBefore(date) {
- var now = moment(date);
- return now.subtract(1, 'hour').valueOf();
- }
+ }
+})();
- function isSameDay(day1, day2) {
- day1 = moment(day1);
- day2 = moment(day2);
+(function() {
+ 'use strict';
- if(day1.startOf('day').isSame(day2.startOf('day'))) {
- return true;
- }
- return false;
- }
+ angular.module('app.components')
+ .controller('StaticController', StaticController);
- function isDiffMoreThan15min(dateToCheckFrom, dateToCheckTo) {
- var duration = moment.duration(moment(dateToCheckTo).diff(moment(dateToCheckFrom)));
- return duration.as('minutes') > 15;
- }
+ StaticController.$inject = ['$timeout', 'animation', '$mdDialog', '$location', '$anchorScroll'];
- function isWithin15min(dateToCheck) {
- var fifteenMinAgo = moment().subtract(15, 'minutes').valueOf();
- dateToCheck = moment(dateToCheck).valueOf();
+ function StaticController($timeout, animation, $mdDialog, $location, $anchorScroll) {
+ var vm = this;
- return dateToCheck > fifteenMinAgo;
- }
+ vm.showStore = showStore;
- function isWithin1Month(dateToCheck) {
- var oneMonthAgo = moment().subtract(1, 'months').valueOf();
- dateToCheck = moment(dateToCheck).valueOf();
+ $anchorScroll.yOffset = 80;
- return dateToCheck > oneMonthAgo;
- }
+ ///////////////////////
- function isWithin(number, type, dateToCheck) {
- var ago = moment().subtract(number, type).valueOf();
- dateToCheck = moment(dateToCheck).valueOf();
+ initialize();
- return dateToCheck > ago;
+ //////////////////
+
+ function initialize() {
+ $timeout(function() {
+ animation.viewLoaded();
+ if($location.hash()){
+ $anchorScroll();
+ }
+ }, 500);
}
- function parseDate(object){
- var time = object;
- return {
- raw: time,
- parsed: !time ? 'No time' : moment(time).format('MMMM DD, YYYY - HH:mm'),
- ago: !time ? 'No time' : moment(time).fromNow()
- }
+ function showStore() {
+ $mdDialog.show({
+ hasBackdrop: true,
+ controller: 'StoreModalController',
+ templateUrl: 'app/components/store/storeModal.html',
+ clickOutsideToClose: true
+ });
}
}
})();
@@ -4953,211 +4989,226 @@ function cookiesLaw($cookies) {
'use strict';
angular.module('app.components')
- .factory('sensorUtils', sensorUtils);
+ .controller('SignupModalController', SignupModalController);
- sensorUtils.$inject = ['timeUtils'];
- function sensorUtils(timeUtils) {
- var service = {
- getRollup: getRollup,
- getSensorName: getSensorName,
- getSensorValue: getSensorValue,
- getSensorPrevValue: getSensorPrevValue,
- getSensorIcon: getSensorIcon,
- getSensorArrow: getSensorArrow,
- getSensorColor: getSensorColor,
- getSensorDescription: getSensorDescription
+ SignupModalController.$inject = ['$scope', '$mdDialog', 'user',
+ 'alert', 'animation'];
+ function SignupModalController($scope, $mdDialog, user,
+ alert, animation ) {
+ var vm = this;
+ vm.answer = function(signupForm) {
+
+ if (!signupForm.$valid){
+ return;
+ }
+
+ $scope.waitingFromServer = true;
+ user.createUser(vm.user)
+ .then(function() {
+ alert.success('Signup was successful');
+ $mdDialog.hide();
+ }).catch(function(err) {
+ alert.error('Signup failed');
+ $scope.errors = err.data.errors;
+ })
+ .finally(function() {
+ $scope.waitingFromServer = false;
+ });
+ };
+ $scope.hide = function() {
+ $mdDialog.hide();
+ };
+ $scope.cancel = function() {
+ $mdDialog.cancel();
};
- return service;
- ///////////////
+ $scope.openLogin = function() {
+ animation.showLogin();
+ $mdDialog.hide();
+ };
+ }
+})();
- function getRollup(dateFrom, dateTo) {
+(function() {
+ 'use strict';
- // Calculate how many data points we can fit on a users screen
- // Smaller screens request less data from the API
- var durationInSec = moment(dateTo).diff(moment(dateFrom)) / 1000;
- var chartWidth = window.innerWidth / 2;
+ angular.module('app.components')
+ .directive('signup', signup);
- var rollup = parseInt(durationInSec / chartWidth) + 's';
+ function signup() {
+ return {
+ scope: {
+ show: '=',
+ },
+ restrict: 'A',
+ controller: 'SignupController',
+ controllerAs: 'vm',
+ templateUrl: 'app/components/signup/signup.html'
+ };
+ }
+})();
- /*
- //var rangeDays = timeUtils.getCurrentRange(dateFrom, dateTo, {format: 'd'});
- var rollup;
- if(rangeDays <= 1) {
- rollup = '15s';
- } else if(rangeDays <= 7) {
- rollup = '1h';//rollup = '15m';
- } else if(rangeDays > 7) {
- rollup = '1d';
- }
- */
- return rollup;
- }
+(function() {
+ 'use strict';
- function getSensorName(name) {
+ angular.module('app.components')
+ .controller('SignupController', SignupController);
- var sensorName;
- // TODO: Improvement check how we set new names
- if( new RegExp('custom circuit', 'i').test(name) ) {
- sensorName = name;
- } else {
- if(new RegExp('noise', 'i').test(name) ) {
- sensorName = 'SOUND';
- } else if(new RegExp('light', 'i').test(name) ) {
- sensorName = 'LIGHT';
- } else if((new RegExp('nets', 'i').test(name) ) ||
- (new RegExp('wifi', 'i').test(name))) {
- sensorName = 'NETWORKS';
- } else if(new RegExp('co', 'i').test(name) ) {
- sensorName = 'CO';
- } else if(new RegExp('no2', 'i').test(name) ) {
- sensorName = 'NO2';
- } else if(new RegExp('humidity', 'i').test(name) ) {
- sensorName = 'HUMIDITY';
- } else if(new RegExp('temperature', 'i').test(name) ) {
- sensorName = 'TEMPERATURE';
- } else if(new RegExp('panel', 'i').test(name) ) {
- sensorName = 'SOLAR PANEL';
- } else if(new RegExp('battery', 'i').test(name) ) {
- sensorName = 'BATTERY';
- } else if(new RegExp('barometric pressure', 'i').test(name) ) {
- sensorName = 'BAROMETRIC PRESSURE';
- } else if(new RegExp('PM 1', 'i').test(name) ) {
- sensorName = 'PM 1';
- } else if(new RegExp('PM 2.5', 'i').test(name) ) {
- sensorName = 'PM 2.5';
- } else if(new RegExp('PM 10', 'i').test(name) ) {
- sensorName = 'PM 10';
- } else {
- sensorName = name;
- }
- }
- return sensorName.toUpperCase();
- }
-
- function getSensorValue(sensor) {
- var value = sensor.value;
+ SignupController.$inject = ['$scope', '$mdDialog'];
+ function SignupController($scope, $mdDialog) {
+ var vm = this;
- if(isNaN(parseInt(value))) {
- value = 'NA';
- } else {
- value = round(value, 1).toString();
- }
+ vm.showSignup = showSignup;
- return value;
- }
+ $scope.$on('showSignup', function() {
+ showSignup();
+ });
+ ////////////////////////
- function round(value, precision) {
- var multiplier = Math.pow(10, precision || 0);
- return Math.round(value * multiplier) / multiplier;
- }
- function getSensorPrevValue(sensor) {
- /*jshint camelcase: false */
- var prevValue = sensor.prev_value;
- return (prevValue && prevValue.toString() ) || 0;
+ function showSignup() {
+ $mdDialog.show({
+ fullscreen: true,
+ hasBackdrop: true,
+ controller: 'SignupModalController',
+ controllerAs: 'vm',
+ templateUrl: 'app/components/signup/signupModal.html',
+ clickOutsideToClose: true
+ });
}
+ }
+})();
- function getSensorIcon(sensorName) {
-
- var thisName = getSensorName(sensorName);
-
- switch(thisName) {
- case 'TEMPERATURE':
- return './assets/images/temperature_icon_new.svg';
-
- case 'HUMIDITY':
- return './assets/images/humidity_icon_new.svg';
+(function() {
+'use strict';
- case 'LIGHT':
- return './assets/images/light_icon_new.svg';
- case 'SOUND':
- return './assets/images/sound_icon_new.svg';
+ angular.module('app.components')
+ .directive('search', search);
- case 'CO':
- return './assets/images/co_icon_new.svg';
+ function search() {
+ return {
+ scope: true,
+ restrict: 'E',
+ templateUrl: 'app/components/search/search.html',
+ controller: 'SearchController',
+ controllerAs: 'vm'
+ };
+ }
+})();
- case 'NO2':
- return './assets/images/no2_icon_new.svg';
+(function() {
+ 'use strict';
- case 'NETWORKS':
- return './assets/images/networks_icon.svg';
+ angular.module('app.components')
+ .controller('SearchController', SearchController);
- case 'BATTERY':
- return './assets/images/battery_icon.svg';
+ SearchController.$inject = ['$scope', 'search', 'SearchResult', '$location', 'animation', 'SearchResultLocation'];
+ function SearchController($scope, search, SearchResult, $location, animation, SearchResultLocation) {
+ var vm = this;
- case 'SOLAR PANEL':
- return './assets/images/solar_panel_icon.svg';
+ vm.searchTextChange = searchTextChange;
+ vm.selectedItemChange = selectedItemChange;
+ vm.querySearch = querySearch;
- case 'BAROMETRIC PRESSURE':
- return './assets/images/pressure_icon_new.svg';
+ ///////////////////
- case 'PM 1':
- case 'PM 2.5':
- case 'PM 10':
- return './assets/images/particle_icon_new.svg';
+ function searchTextChange() {
+ }
- default:
- return './assets/images/unknownsensor_icon.svg';
+ function selectedItemChange(result) {
+ if (!result) { return; }
+ if(result.type === 'User') {
+ $location.path('/users/' + result.id);
+ } else if(result.type === 'Device') {
+ $location.path('/kits/' + result.id);
+ } else if (result.type === 'City'){
+ animation.goToLocation({lat: result.lat, lng: result.lng, type: result.type, layer: result.layer});
}
}
- function getSensorArrow(currentValue, prevValue) {
- currentValue = parseInt(currentValue) || 0;
- prevValue = parseInt(prevValue) || 0;
-
- if(currentValue > prevValue) {
- return 'arrow_up';
- } else if(currentValue < prevValue) {
- return 'arrow_down';
- } else {
- return 'equal';
+ function querySearch(query) {
+ if(query.length < 3) {
+ return [];
}
- }
- function getSensorColor(sensorName) {
- switch(getSensorName(sensorName)) {
- case 'TEMPERATURE':
- return '#FF3D4C';
+ return search.globalSearch(query)
+ .then(function(data) {
- case 'HUMIDITY':
- return '#55C4F5';
+ return data.map(function(object) {
- case 'LIGHT':
- return '#ffc107';
+ if(object.type === 'City' || object.type === 'Country') {
+ return new SearchResultLocation(object);
+ } else {
+ return new SearchResult(object);
+ }
+ });
+ });
+ }
+ }
+})();
- case 'SOUND':
- return '#0019FF';
+(function() {
+ 'use strict';
- case 'CO':
- return '#00A103';
+ angular.module('app.components')
+ .controller('PasswordResetController', PasswordResetController);
- case 'NO2':
- return '#8cc252';
+ PasswordResetController.$inject = ['$mdDialog', '$stateParams', '$timeout',
+ 'animation', '$location', 'alert', 'auth'];
+ function PasswordResetController($mdDialog, $stateParams, $timeout,
+ animation, $location, alert, auth) {
+
+ var vm = this;
+ vm.showForm = false;
+ vm.form = {};
+ vm.isDifferent = false;
+ vm.answer = answer;
- case 'NETWORKS':
- return '#681EBD';
+ initialize();
+ ///////////
- case 'SOLAR PANEL':
- return '#d555ce';
+ function initialize() {
+ $timeout(function() {
+ animation.viewLoaded();
+ }, 500);
+ getUserData();
+ }
- case 'BATTERY':
- return '#ff8601';
+ function getUserData() {
+ auth.getResetPassword($stateParams.code)
+ .then(function() {
+ vm.showForm = true;
+ })
+ .catch(function() {
+ alert.error('Wrong url');
+ $location.path('/');
+ });
+ }
- default:
- return '#0019FF';
+ function answer(data) {
+ vm.waitingFromServer = true;
+ vm.errors = undefined;
+
+ if(data.newPassword === data.confirmPassword) {
+ vm.isDifferent = false;
+ } else {
+ vm.isDifferent = true;
+ return;
}
- }
- function getSensorDescription(sensorID, sensorTypes) {
- return _(sensorTypes)
- .chain()
- .find(function(sensorType) {
- return sensorType.id === sensorID;
+ auth.patchResetPassword($stateParams.code, {password: data.newPassword})
+ .then(function() {
+ alert.success('Your data was updated successfully');
+ $location.path('/profile');
})
- .value()
- .measurement.description;
+ .catch(function(err) {
+ alert.error('Your data wasn\'t updated');
+ vm.errors = err.data.errors;
+ })
+ .finally(function() {
+ vm.waitingFromServer = false;
+ });
}
}
})();
@@ -5166,62 +5217,43 @@ function cookiesLaw($cookies) {
'use strict';
angular.module('app.components')
- .factory('searchUtils', searchUtils);
+ .controller('PasswordRecoveryModalController', PasswordRecoveryModalController);
+ PasswordRecoveryModalController.$inject = ['$scope', 'animation', '$mdDialog', 'auth', 'alert'];
+ function PasswordRecoveryModalController($scope, animation, $mdDialog, auth, alert) {
- searchUtils.$inject = [];
- function searchUtils() {
- var service = {
- parseLocation: parseLocation,
- parseName: parseName,
- parseIcon: parseIcon,
- parseIconType: parseIconType
+ $scope.hide = function() {
+ $mdDialog.hide();
+ };
+ $scope.cancel = function() {
+ $mdDialog.cancel();
};
- return service;
-
- /////////////////
-
- function parseLocation(object) {
- var location = '';
-
- if(!!object.city) {
- location += object.city;
- }
- if(!!object.city && !!object.country) {
- location += ', ';
- }
- if(!!object.country) {
- location += object.country;
- }
-
- return location;
- }
- function parseName(object) {
- var name = object.type === 'User' ? object.username : object.name;
- return name;
- }
+ $scope.recoverPassword = function() {
+ $scope.waitingFromServer = true;
+ var data = {
+ /*jshint camelcase: false */
+ email_or_username: $scope.input
+ };
- function parseIcon(object, type) {
- switch(type) {
- case 'User':
- return object.profile_picture;
- case 'Device':
- return 'assets/images/kit.svg';
- case 'Country':
- case 'City':
- return 'assets/images/location_icon_normal.svg';
- }
- }
+ auth.recoverPassword(data)
+ .then(function() {
+ alert.success('You were sent an email to recover your password');
+ $mdDialog.hide();
+ })
+ .catch(function(err) {
+ alert.error('That username doesn\'t exist');
+ $scope.errors = err.data;
+ })
+ .finally(function() {
+ $scope.waitingFromServer = false;
+ });
+ };
- function parseIconType(type) {
- switch(type) {
- case 'Device':
- return 'div';
- default:
- return 'img';
- }
- }
+ $scope.openSignup = function() {
+ animation.showSignup();
+ $mdDialog.hide();
+ };
}
})();
@@ -5229,360 +5261,424 @@ function cookiesLaw($cookies) {
'use strict';
angular.module('app.components')
- .factory('markerUtils', markerUtils);
-
- markerUtils.$inject = ['deviceUtils', 'MARKER_ICONS'];
- function markerUtils(deviceUtils, MARKER_ICONS) {
- var service = {
- getIcon: getIcon,
- getMarkerIcon: getMarkerIcon,
- };
- _.defaults(service, deviceUtils);
- return service;
-
- ///////////////
+ .controller('PasswordRecoveryController', PasswordRecoveryController);
- function getIcon(object) {
- var icon;
- var labels = deviceUtils.parseSystemTags(object);
- var isSCKHardware = deviceUtils.isSCKHardware(object);
+ PasswordRecoveryController.$inject = ['auth', 'alert', '$mdDialog'];
+ function PasswordRecoveryController(auth, alert, $mdDialog) {
+ var vm = this;
- if(hasLabel(labels, 'offline')) {
- icon = MARKER_ICONS.markerSmartCitizenOffline;
- } else if (isSCKHardware) {
- icon = MARKER_ICONS.markerSmartCitizenOnline;
- } else {
- icon = MARKER_ICONS.markerExperimentalNormal;
- }
- return icon;
- }
+ vm.waitingFromServer = false;
+ vm.errors = undefined;
+ vm.recoverPassword = recoverPassword;
- function hasLabel(labels, targetLabel) {
- return _.some(labels, function(label) {
- return label === targetLabel;
- });
- }
+ ///////////////
- function getMarkerIcon(marker, state) {
- var markerType = marker.icon.className;
+ function recoverPassword() {
+ vm.waitingFromServer = true;
+ vm.errors = undefined;
+
+ var data = {
+ username: vm.username
+ };
- if(state === 'active') {
- marker.icon = MARKER_ICONS[markerType + 'Active'];
- marker.focus = true;
- } else if(state === 'inactive') {
- var targetClass = markerType.split(' ')[0];
- marker.icon = MARKER_ICONS[targetClass];
- }
- return marker;
+ auth.recoverPassword(data)
+ .then(function() {
+ alert.success('You were sent an email to recover your password');
+ $mdDialog.hide();
+ })
+ .catch(function(err) {
+ vm.errors = err.data.errors;
+ if(vm.errors) {
+ alert.error('That email/username doesn\'t exist');
+ }
+ })
+ .finally(function() {
+ vm.waitingFromServer = false;
+ });
}
- }
+ }
})();
(function() {
'use strict';
angular.module('app.components')
- .factory('mapUtils', mapUtils);
+ .controller('MyProfileController', MyProfileController);
- mapUtils.$inject = [];
- function mapUtils() {
- var service = {
- getDefaultFilters: getDefaultFilters,
- setDefaultFilters: setDefaultFilters,
- canFilterBeRemoved: canFilterBeRemoved
- };
- return service;
+ MyProfileController.$inject = ['$scope', '$location', '$q', '$interval',
+ 'userData', 'AuthUser', 'user', 'auth', 'alert',
+ 'COUNTRY_CODES', '$timeout', 'file', 'animation',
+ '$mdDialog', 'PreviewDevice', 'device', 'deviceUtils',
+ 'userUtils', '$filter', '$state', 'Restangular', '$window'];
+ function MyProfileController($scope, $location, $q, $interval,
+ userData, AuthUser, user, auth, alert,
+ COUNTRY_CODES, $timeout, file, animation,
+ $mdDialog, PreviewDevice, device, deviceUtils,
+ userUtils, $filter, $state, Restangular, $window) {
- //////////////
+ var vm = this;
- function getDefaultFilters(filterData, defaultFilters) {
- var obj = {};
- if(!filterData.indoor && !filterData.outdoor) {
- obj[defaultFilters.exposure] = true;
- }
- if(!filterData.online && !filterData.offline) {
- obj[defaultFilters.status] = true;
- }
- return obj;
- }
+ vm.unhighlightIcon = unhighlightIcon;
- function setDefaultFilters(filterData) {
- var obj = {};
- if(!filterData.indoor || !filterData.outdoor) {
- obj.exposure = filterData.indoor ? 'indoor' : 'outdoor';
- }
- if(!filterData.online || !filterData.offline) {
- obj.status = filterData.online ? 'online' : 'offline';
- }
- return obj;
- }
+ //PROFILE TAB
+ vm.formUser = {};
+ vm.getCountries = getCountries;
- function canFilterBeRemoved(filterData, filterName) {
- if(filterName === 'indoor' || filterName === 'outdoor') {
- return filterData.indoor && filterData.outdoor;
- } else if(filterName === 'online' || filterName === 'offline') {
- return filterData.online && filterData.offline;
- }
- }
- }
-})();
+ vm.user = userData;
+ copyUserToForm(vm.formUser, vm.user);
+ vm.searchText = vm.formUser.country;
-(function() {
- 'use strict';
- angular.module('app.components')
- .config(function ($provide) {
- $provide.decorator('$exceptionHandler', ['$delegate', function($delegate) {
- return function (exception, cause) {
- /*jshint camelcase: false */
- $delegate(exception, cause);
- };
- }]);
+ vm.updateUser = updateUser;
+ vm.removeUser = removeUser;
+ vm.uploadAvatar = uploadAvatar;
- });
-})();
+ //THIS IS TEMPORARY.
+ // Will grow on to a dynamic API KEY management
+ // with the new /accounts oAuth mgmt methods
-(function() {
- 'use strict';
+ // The auth controller has not populated the `user` at this point,
+ // so user.token is undefined
+ // This controller depends on auth has already been run.
+ vm.user.token = auth.getToken();
+ vm.addNewDevice = addNewDevice;
- angular.module('app.components')
- .factory('deviceUtils', deviceUtils);
+ //KITS TAB
+ vm.devices = [];
+ vm.deviceStatus = undefined;
+ vm.removeDevice = removeDevice;
+ vm.downloadData = downloadData;
- deviceUtils.$inject = ['COUNTRY_CODES', 'device'];
- function deviceUtils(COUNTRY_CODES, device) {
- var service = {
- parseLocation: parseLocation,
- parseCoordinates: parseCoordinates,
- parseSystemTags: parseSystemTags,
- parseUserTags: parseUserTags,
- classify: classify,
- parseNotifications: parseNotifications,
- parseOwner: parseOwner,
- parseName: parseName,
- parseString: parseString,
- parseHardware: parseHardware,
- parseHardwareInfo: parseHardwareInfo,
- parseHardwareName: parseHardwareName,
- isPrivate: isPrivate,
- preciseLocation: preciseLocation,
- enableForwarding: enableForwarding,
- isLegacyVersion: isLegacyVersion,
- isSCKHardware: isSCKHardware,
- parseState: parseState,
- parseAvatar: parseAvatar,
- belongsToUser: belongsToUser,
- parseSensorTime: parseSensorTime
- };
+ vm.filteredDevices = [];
+ vm.dropdownSelected = undefined;
- return service;
+ //SIDEBAR
+ vm.filterDevices = filterDevices;
+ vm.filterTools = filterTools;
- ///////////////
+ vm.selectThisTab = selectThisTab;
- function parseLocation(object) {
- var location = '';
- var city = '';
- var country = '';
+ $scope.$on('loggedOut', function() {
+ $location.path('/');
+ });
- if (object.location) {
- city = object.location.city;
- country = object.location.country;
- if(!!city) {
- location += city;
- }
- if(!!city && !!location) {
- location += ', '
- }
- if(!!country) {
- location += country;
- }
+ $scope.$on('devicesContextUpdated', function(){
+ var userData = auth.getCurrentUser().data;
+ if(userData){
+ vm.user = userData;
}
- return location;
- }
+ initialize();
+ });
- function parseCoordinates(object) {
- if (object.location) {
- return {
- lat: object.location.latitude,
- lng: object.location.longitude
- };
- }
- // TODO: Bug - what happens if no location?
- }
+ initialize();
- function parseSystemTags(object) {
- /*jshint camelcase: false */
- return object.system_tags;
- }
+ //////////////////
- function parseUserTags(object) {
- return object.user_tags;
- }
+ function initialize() {
+
+ startingTab();
+ if(!vm.user.devices.length) {
+ vm.devices = [];
+ animation.viewLoaded();
+ } else {
+
+ vm.devices = vm.user.devices.map(function(data) {
+ return new PreviewDevice(data);
+ })
+
+ $timeout(function() {
+ mapWithBelongstoUser(vm.devices);
+ filterDevices(vm.status);
+ setSidebarMinHeight();
+ animation.viewLoaded();
+ });
- function parseNotifications(object){
- return {
- lowBattery: object.notify.low_battery,
- stopPublishing: object.notify.stopped_publishing
}
}
- function classify(kitType) {
- if(!kitType) {
- return '';
+ function filterDevices(status) {
+ if(status === 'all') {
+ status = undefined;
}
- return kitType.toLowerCase().split(' ').join('_');
+ vm.deviceStatus = status;
+ vm.filteredDevices = $filter('filterLabel')(vm.devices, vm.deviceStatus);
}
- function parseName(object, trim=false) {
- if(!object.name) {
- return;
- }
- if (trim) {
- return object.name.length <= 41 ? object.name : object.name.slice(0, 35).concat(' ... ');
+ function filterTools(type) {
+ if(type === 'all') {
+ type = undefined;
}
- return object.name;
+ vm.toolType = type;
}
- function parseHardware(object) {
- if (!object.hardware) {
- return;
+ function updateUser(userData) {
+ if(userData.country) {
+ _.each(COUNTRY_CODES, function(value, key) {
+ if(value === userData.country) {
+ /*jshint camelcase: false */
+ userData.country_code = key;
+ return;
+ }
+ });
+ } else {
+ userData.country_code = null;
}
- return {
- name: parseString(object.hardware.name),
- type: parseString(object.hardware.type),
- description: parseString(object.hardware.description),
- version: parseVersionString(object.hardware.version),
- slug: object.hardware.slug,
- info: parseHardwareInfo(object.hardware.info)
- }
+ user.updateUser(userData)
+ .then(function(data) {
+ var user = new AuthUser(data);
+ _.extend(vm.user, user);
+ auth.updateUser();
+ vm.errors = {};
+ alert.success('User updated');
+ })
+ .catch(function(err) {
+ alert.error('User could not be updated ');
+ vm.errors = err.data.errors;
+ });
}
- function parseString(str) {
- if (typeof(str) !== 'string') { return null; }
- return str;
- }
+ function removeUser() {
+ var confirm = $mdDialog.confirm()
+ .title('Delete your account?')
+ .textContent('Are you sure you want to delete your account?')
+ .ariaLabel('')
+ .ok('delete')
+ .cancel('cancel')
+ .theme('primary')
+ .clickOutsideToClose(true);
- function parseVersionString (str) {
- if (typeof(str) !== 'string') { return null; }
- var x = str.split('.');
- // parse from string or default to 0 if can't parse
- var maj = parseInt(x[0]) || 0;
- var min = parseInt(x[1]) || 0;
- var pat = parseInt(x[2]) || 0;
- return {
- major: maj,
- minor: min,
- patch: pat
- };
+ $mdDialog.show(confirm)
+ .then(function(){
+ return Restangular.all('').customDELETE('me')
+ .then(function(){
+ alert.success('Account removed successfully. Redirecting you…');
+ $timeout(function(){
+ auth.logout();
+ $state.transitionTo('landing');
+ }, 2000);
+ })
+ .catch(function(){
+ alert.error('Error occurred trying to delete your account.');
+ });
+ });
}
- function parseHardwareInfo (object) {
- if (!object) { return null; } // null
- if (typeof(object) == 'string') { return null; } // FILTERED
-
- var id = parseString(object.id);
- var mac = parseString(object.mac);
- var time = Date(object.time);
- var esp_bd = parseString(object.esp_bd);
- var hw_ver = parseString(object.hw_ver);
- var sam_bd = parseString(object.sam_bd);
- var esp_ver = parseString(object.esp_ver);
- var sam_ver = parseString(object.sam_ver);
+ function selectThisTab(iconIndex, uistate){
+ /* This looks more like a hack but we need to workout how to properly use md-tab with ui-router */
- return {
- id: id,
- mac: mac,
- time: time,
- esp_bd: esp_bd,
- hw_ver: hw_ver,
- sam_bd: sam_bd,
- esp_ver: esp_ver,
- sam_ver: sam_ver
- };
- }
+ highlightIcon(iconIndex);
- function parseHardwareName(object) {
- if (object.hasOwnProperty('hardware')) {
- if (!object.hardware.name) {
- return 'Unknown hardware'
- }
- return object.hardware.name;
+ if ($state.current.name.includes('myProfileAdmin')){
+ var transitionState = 'layout.myProfileAdmin.' + uistate;
+ $state.transitionTo(transitionState, {id: userData.id});
} else {
- return 'Unknown hardware'
+ var transitionState = 'layout.myProfile.' + uistate;
+ $state.transitionTo(transitionState);
}
- }
- function isPrivate(object) {
- return object.data_policy.is_private;
}
- function preciseLocation(object) {
- return object.data_policy.precise_location;
- }
+ function startingTab() {
+ /* This looks more like a hack but we need to workout how to properly use md-tab with ui-router */
- function enableForwarding(object) {
- return object.data_policy.enable_forwarding ;
- }
+ var childState = $state.current.name.split('.').pop();
- function isLegacyVersion (object) {
- if (!object.hardware || !object.hardware.version || object.hardware.version.major > 1) {
- return false;
- } else {
- if (object.hardware.version.major == 1 && object.hardware.version.minor <5 ){
- return true;
- }
- return false;
+ switch(childState) {
+ case 'user':
+ vm.startingTab = 1;
+ break;
+ default:
+ vm.startingTab = 0;
+ break;
}
- }
- function isSCKHardware (object){
- if (!object.hardware || !object.hardware.type || object.hardware.type != 'SCK') {
- return false;
- } else {
- return true;
- }
}
- function parseOwner(object) {
- return {
- id: object.owner.id,
- username: object.owner.username,
- /*jshint camelcase: false */
- devices: object.owner.device_ids,
- city: object.owner.location.city,
- country: COUNTRY_CODES[object.owner.location.country_code],
- url: object.owner.url,
- profile_picture: object.owner.profile_picture
- };
+ function highlightIcon(iconIndex) {
+
+ var icons = angular.element('.myProfile_tab_icon');
+
+ _.each(icons, function(icon) {
+ unhighlightIcon(icon);
+ });
+
+ var icon = icons[iconIndex];
+
+ angular.element(icon).find('.stroke_container').css({'stroke': 'white', 'stroke-width': '0.01px'});
+ angular.element(icon).find('.fill_container').css('fill', 'white');
}
- function parseState(status) {
- var name = parseStateName(status);
- var className = classify(name);
+ function unhighlightIcon(icon) {
+ icon = angular.element(icon);
- return {
- name: name,
- className: className
+ icon.find('.stroke_container').css({'stroke': 'none'});
+ icon.find('.fill_container').css('fill', '#FF8600');
+ }
+
+ function setSidebarMinHeight() {
+ var height = document.body.clientHeight / 4 * 3;
+ angular.element('.profile_content').css('min-height', height + 'px');
+ }
+
+ function getCountries(searchText) {
+ return _.filter(COUNTRY_CODES, createFilter(searchText));
+ }
+
+ function createFilter(searchText) {
+ searchText = searchText.toLowerCase();
+ return function(country) {
+ country = country.toLowerCase();
+ return country.indexOf(searchText) !== -1;
};
}
- function parseStateName(object) {
- return object.state.replace('_', ' ');
+ function uploadAvatar(fileData) {
+ if(fileData && fileData.length) {
+
+ // TODO: Improvement Is there a simpler way to patch the image to the API and use the response?
+ // Something like:
+ //Restangular.all('/me').patch(data);
+ // Instead of doing it manually like here:
+ var fd = new FormData();
+ fd.append('profile_picture', fileData[0]);
+ Restangular.one('/me')
+ .withHttpConfig({transformRequest: angular.identity})
+ .customPATCH(fd, '', undefined, {'Content-Type': undefined})
+ .then(function(resp){
+ vm.user.profile_picture = resp.profile_picture;
+ })
+ }
}
- function parseAvatar() {
- return './assets/images/sckit_avatar.jpg';
+ function copyUserToForm(formData, userData) {
+ var props = {username: true, email: true, city: true, country: true, country_code: true, url: true, constructor: false};
+
+ for(var key in userData) {
+ if(props[key]) {
+ formData[key] = userData[key];
+ }
+ }
}
- function parseSensorTime(sensor) {
- /*jshint camelcase: false */
- return moment(sensor.recorded_at).format('');
+ function mapWithBelongstoUser(devices){
+ _.map(devices, addBelongProperty);
}
- function belongsToUser(devicesArray, deviceID) {
- return _.some(devicesArray, function(device) {
- return device.id === deviceID;
+ function addBelongProperty(device){
+ device.belongProperty = deviceBelongsToUser(device);
+ return device;
+ }
+
+
+ function deviceBelongsToUser(device){
+ if(!auth.isAuth() || !device || !device.id) {
+ return false;
+ }
+ var deviceID = parseInt(device.id);
+ var userData = ( auth.getCurrentUser().data ) ||
+ ($window.localStorage.getItem('smartcitizen.data') &&
+ new AuthUser( JSON.parse(
+ $window.localStorage.getItem('smartcitizen.data') )));
+
+ var belongsToUser = deviceUtils.belongsToUser(userData.devices, deviceID);
+ var isAdmin = userUtils.isAdmin(userData);
+
+ return isAdmin || belongsToUser;
+ }
+
+ function downloadData(device){
+ $mdDialog.show({
+ hasBackdrop: true,
+ controller: 'DownloadModalController',
+ controllerAs: 'vm',
+ templateUrl: 'app/components/download/downloadModal.html',
+ clickOutsideToClose: true,
+ locals: {thisDevice:device}
+ }).then(function(){
+ var alert = $mdDialog.alert()
+ .title('SUCCESS')
+ .textContent('We are processing your data. Soon you will be notified in your inbox')
+ .ariaLabel('')
+ .ok('OK!')
+ .theme('primary')
+ .clickOutsideToClose(true);
+
+ $mdDialog.show(alert);
+ }).catch(function(err){
+ if (!err){
+ return;
+ }
+ var errorAlert = $mdDialog.alert()
+ .title('ERROR')
+ .textContent('Uh-oh, something went wrong')
+ .ariaLabel('')
+ .ok('D\'oh')
+ .theme('primary')
+ .clickOutsideToClose(false);
+
+ $mdDialog.show(errorAlert);
});
}
+
+ function removeDevice(deviceID) {
+ var confirm = $mdDialog.confirm()
+ .title('Delete this kit?')
+ .textContent('Are you sure you want to delete this kit?')
+ .ariaLabel('')
+ .ok('DELETE')
+ .cancel('Cancel')
+ .theme('primary')
+ .clickOutsideToClose(true);
+
+ $mdDialog
+ .show(confirm)
+ .then(function(){
+ device
+ .removeDevice(deviceID)
+ .then(function(){
+ alert.success('Your kit was deleted successfully');
+ device.updateContext();
+ })
+ .catch(function(){
+ alert.error('Error trying to delete your kit.');
+ });
+ });
+ }
+
+ $scope.addDeviceSelector = addDeviceSelector;
+ function addDeviceSelector(){
+ $mdDialog.show({
+ templateUrl: 'app/components/myProfile/addDeviceSelectorModal.html',
+ clickOutsideToClose: true,
+ multiple: true,
+ controller: DialogController,
+ });
+ }
+
+ function DialogController($scope, $mdDialog){
+ $scope.cancel = function(){
+ $mdDialog.cancel();
+ };
+ }
+
+ function addNewDevice() {
+ var confirm = $mdDialog.confirm()
+ .title('Hey! Do you want to add a new kit?')
+ .textContent('Please, notice this currently supports just the SCK 1.0 and SCK 1.1')
+ .ariaLabel('')
+ .ok('Ok')
+ .cancel('Cancel')
+ .theme('primary')
+ .clickOutsideToClose(true);
+
+ $mdDialog
+ .show(confirm)
+ .then(function(){
+ $state.go('layout.kitAdd');
+ });
+ }
+
+
}
})();
@@ -5590,725 +5686,640 @@ function cookiesLaw($cookies) {
'use strict';
angular.module('app.components')
- .filter('filterLabel', filterLabel);
+ .controller('MapTagModalController', MapTagModalController);
+ MapTagModalController.$inject = ['$mdDialog', 'tag', 'selectedTags'];
- function filterLabel() {
- return function(devices, targetLabel) {
- if(targetLabel === undefined) {
- return devices;
- }
- if(devices) {
- return _.filter(devices, function(device) {
- var containsLabel = device.systemTags.indexOf(targetLabel) !== -1;
- if(containsLabel) {
- return containsLabel;
- }
- // This should be fixed or polished in the future
- // var containsNewIfTargetIsOnline = targetLabel === 'online' && _.some(kit.labels, function(label) {return label.indexOf('new') !== -1;});
- // return containsNewIfTargetIsOnline;
- });
- }
- };
+ function MapTagModalController($mdDialog, tag, selectedTags) {
+
+ var vm = this;
+
+ vm.checks = {};
+
+ vm.answer = answer;
+ vm.hide = hide;
+ vm.clear = clear;
+ vm.cancel = cancel;
+ vm.tags = [];
+
+ init();
+
+ ////////////////////////////////////////////////////////
+
+ function init() {
+ tag.getTags()
+ .then(function(tags) {
+ vm.tags = tags;
+
+ _.forEach(selectedTags, select);
+
+ });
}
-})();
-(function() {
- 'use strict';
+ function answer() {
- /**
- * Tools links for user profile
- * @constant
- * @type {Array}
- */
+ var selectedTags = _(vm.tags)
+ .filter(isTagSelected)
+ .value();
+ $mdDialog.hide(selectedTags);
+ }
- angular.module('app.components')
- .constant('PROFILE_TOOLS', [{
- type: 'documentation',
- title: 'How to connect your Smart Citizen Kit tutorial',
- description: 'Adding a Smart Citizen Kit tutorial',
- avatar: '',
- href: 'http://docs.smartcitizen.me/#/start/adding-a-smart-citizen-kit'
- }, {
- type: 'documentation',
- title: 'Download the latest Smart Citizen Kit Firmware',
- description: 'The latest Arduino firmware for your kit',
- avatar: '',
- href: 'https://github.com/fablabbcn/Smart-Citizen-Kit/releases/latest'
- }, {
- type: 'documentation',
- title: 'API Documentation',
- description: 'Documentation for the new API',
- avatar: '',
- href: 'http://developer.smartcitizen.me/'
- }, {
- type: 'community',
- title: 'Smart Citizen Forum',
- description: 'Join the community discussion. Your feedback is important for us.',
- avatar: '',
- href:'http://forum.smartcitizen.me/'
- }, {
- type: 'documentation',
- title: 'Smart Citizen Kit hardware details',
- description: 'Visit the docs',
- avatar: 'https://docs.smartcitizen.me/#/start/hardware'
- }, {
- type: 'documentation',
- title: 'Style Guide',
- description: 'Guidelines of the Smart Citizen UI',
- avatar: '',
- href: '/styleguide'
- }, {
- type: 'social',
- title: 'Like us on Facebook',
- description: 'Join the community on Facebook',
- avatar: '',
- href: 'https://www.facebook.com/smartcitizenBCN'
- }, {
- type: 'social',
- title: 'Follow us on Twitter',
- description: 'Follow our news on Twitter',
- avatar: '',
- href: 'https://twitter.com/SmartCitizenKit'
- }]);
-})();
+ function hide() {
+ answer();
+ }
-(function() {
- 'use strict';
+ function clear() {
+ $mdDialog.hide(null);
+ }
- /**
- * Marker icons
- * @constant
- * @type {Object}
- */
+ function cancel() {
+ answer();
+ }
- angular.module('app.components')
- .constant('MARKER_ICONS', {
- defaultIcon: {},
- markerSmartCitizenNormal: {
- type: 'div',
- className: 'markerSmartCitizenNormal',
- iconSize: [24, 24]
- },
- markerExperimentalNormal: {
- type: 'div',
- className: 'markerExperimentalNormal',
- iconSize: [24, 24]
- },
- markerSmartCitizenOnline: {
- type: 'div',
- className: 'markerSmartCitizenOnline',
- iconSize: [24, 24]
- },
- markerSmartCitizenOnlineActive: {
- type: 'div',
- className: 'markerSmartCitizenOnline marker_blink',
- iconSize: [24, 24]
- },
- markerSmartCitizenOffline: {
- type: 'div',
- className: 'markerSmartCitizenOffline',
- iconSize: [24, 24]
- },
- markerSmartCitizenOfflineActive: {
- type: 'div',
- className: 'markerSmartCitizenOffline marker_blink',
- iconSize: [24, 24]
- }
- });
+ function isTagSelected(tag) {
+ return vm.checks[tag.name];
+ }
+
+ function select(tag){
+ vm.checks[tag] = true;
+ }
+ }
})();
(function() {
'use strict';
- /**
- * Dropdown options for user
- * @constant
- * @type {Array}
- */
angular.module('app.components')
- .constant('DROPDOWN_OPTIONS_USER', [
- {divider: true, text: 'Hi,', href: './profile'},
- {text: 'My profile', href: './profile'},
- {text: 'Log out', href: './logout'}
- ]);
-})();
+ .controller('MapFilterModalController', MapFilterModalController);
-(function() {
- 'use strict';
+ MapFilterModalController.$inject = ['$mdDialog','selectedFilters', '$timeout'];
- /**
- * Dropdown options for community button
- * @constant
- * @type {Array}
- */
+ function MapFilterModalController($mdDialog, selectedFilters, $timeout) {
- angular.module('app.components')
- .constant('DROPDOWN_OPTIONS_COMMUNITY', [
- {text: 'About', href: '/about'},
- {text: 'Forum', href: 'https://forum.smartcitizen.me/'},
- {text: 'Documentation', href: 'http://docs.smartcitizen.me/'},
- {text: 'API Reference', href: 'http://developer.smartcitizen.me/'},
- {text: 'Github', href: 'https://github.com/fablabbcn/Smart-Citizen-Kit'},
- {text: 'Legal', href: '/policy'}
- ]);
+ var vm = this;
+
+ vm.checks = {};
+
+ vm.answer = answer;
+ vm.hide = hide;
+ vm.clear = clear;
+ vm.cancel = cancel;
+ vm.toggle = toggle;
+
+ vm.location = ['indoor', 'outdoor'];
+ vm.status = ['online', 'offline'];
+ vm.new = ['new'];
+
+ vm.filters = [];
+
+ init();
+
+ ////////////////////////////////////////////////////////
+
+ function init() {
+ _.forEach(selectedFilters, select);
+ }
+
+ function answer() {
+ vm.filters = vm.filters.concat(vm.location, vm.status, vm.new);
+ var selectedFilters = _(vm.filters)
+ .filter(isFilterSelected)
+ .value();
+ $mdDialog.hide(selectedFilters);
+ }
+
+ function hide() {
+ answer();
+ }
+
+ function clear() {
+ vm.filters = vm.filters.concat(vm.location, vm.status, vm.new);
+ $mdDialog.hide(vm.filters);
+ }
+
+ function cancel() {
+ answer();
+ }
+
+ function isFilterSelected(filter) {
+ return vm.checks[filter];
+ }
+
+ function toggle(filters) {
+ $timeout(function() {
+
+ for (var i = 0; i < filters.length - 1; i++) {
+ if (vm.checks[filters[i]] === false && vm.checks[filters[i]] === vm.checks[filters[i+1]]) {
+ for (var n = 0; n < filters.length; n++) {
+ vm.checks[filters[n]] = true;
+ }
+ }
+ }
+
+ });
+ }
+
+ function select(filter){
+ vm.checks[filter] = true;
+ }
+ }
})();
(function() {
'use strict';
- /**
- * Country codes.
- * @constant
- * @type {Object}
- */
-
angular.module('app.components')
- .constant('COUNTRY_CODES', {
- 'AF': 'Afghanistan',
- 'AX': 'Aland Islands',
- 'AL': 'Albania',
- 'DZ': 'Algeria',
- 'AS': 'American Samoa',
- 'AD': 'Andorra',
- 'AO': 'Angola',
- 'AI': 'Anguilla',
- 'AQ': 'Antarctica',
- 'AG': 'Antigua And Barbuda',
- 'AR': 'Argentina',
- 'AM': 'Armenia',
- 'AW': 'Aruba',
- 'AU': 'Australia',
- 'AT': 'Austria',
- 'AZ': 'Azerbaijan',
- 'BS': 'Bahamas',
- 'BH': 'Bahrain',
- 'BD': 'Bangladesh',
- 'BB': 'Barbados',
- 'BY': 'Belarus',
- 'BE': 'Belgium',
- 'BZ': 'Belize',
- 'BJ': 'Benin',
- 'BM': 'Bermuda',
- 'BT': 'Bhutan',
- 'BO': 'Bolivia',
- 'BA': 'Bosnia And Herzegovina',
- 'BW': 'Botswana',
- 'BV': 'Bouvet Island',
- 'BR': 'Brazil',
- 'IO': 'British Indian Ocean Territory',
- 'BN': 'Brunei Darussalam',
- 'BG': 'Bulgaria',
- 'BF': 'Burkina Faso',
- 'BI': 'Burundi',
- 'KH': 'Cambodia',
- 'CM': 'Cameroon',
- 'CA': 'Canada',
- 'CV': 'Cape Verde',
- 'KY': 'Cayman Islands',
- 'CF': 'Central African Republic',
- 'TD': 'Chad',
- 'CL': 'Chile',
- 'CN': 'China',
- 'CX': 'Christmas Island',
- 'CC': 'Cocos (Keeling) Islands',
- 'CO': 'Colombia',
- 'KM': 'Comoros',
- 'CG': 'Congo',
- 'CD': 'Congo, Democratic Republic',
- 'CK': 'Cook Islands',
- 'CR': 'Costa Rica',
- 'CI': 'Cote D\'Ivoire',
- 'HR': 'Croatia',
- 'CU': 'Cuba',
- 'CY': 'Cyprus',
- 'CZ': 'Czech Republic',
- 'DK': 'Denmark',
- 'DJ': 'Djibouti',
- 'DM': 'Dominica',
- 'DO': 'Dominican Republic',
- 'EC': 'Ecuador',
- 'EG': 'Egypt',
- 'SV': 'El Salvador',
- 'GQ': 'Equatorial Guinea',
- 'ER': 'Eritrea',
- 'EE': 'Estonia',
- 'ET': 'Ethiopia',
- 'FK': 'Falkland Islands (Malvinas)',
- 'FO': 'Faroe Islands',
- 'FJ': 'Fiji',
- 'FI': 'Finland',
- 'FR': 'France',
- 'GF': 'French Guiana',
- 'PF': 'French Polynesia',
- 'TF': 'French Southern Territories',
- 'GA': 'Gabon',
- 'GM': 'Gambia',
- 'GE': 'Georgia',
- 'DE': 'Germany',
- 'GH': 'Ghana',
- 'GI': 'Gibraltar',
- 'GR': 'Greece',
- 'GL': 'Greenland',
- 'GD': 'Grenada',
- 'GP': 'Guadeloupe',
- 'GU': 'Guam',
- 'GT': 'Guatemala',
- 'GG': 'Guernsey',
- 'GN': 'Guinea',
- 'GW': 'Guinea-Bissau',
- 'GY': 'Guyana',
- 'HT': 'Haiti',
- 'HM': 'Heard Island & Mcdonald Islands',
- 'VA': 'Holy See (Vatican City State)',
- 'HN': 'Honduras',
- 'HK': 'Hong Kong',
- 'HU': 'Hungary',
- 'IS': 'Iceland',
- 'IN': 'India',
- 'ID': 'Indonesia',
- 'IR': 'Iran, Islamic Republic Of',
- 'IQ': 'Iraq',
- 'IE': 'Ireland',
- 'IM': 'Isle Of Man',
- 'IL': 'Israel',
- 'IT': 'Italy',
- 'JM': 'Jamaica',
- 'JP': 'Japan',
- 'JE': 'Jersey',
- 'JO': 'Jordan',
- 'KZ': 'Kazakhstan',
- 'KE': 'Kenya',
- 'KI': 'Kiribati',
- 'KR': 'Korea',
- 'KW': 'Kuwait',
- 'KG': 'Kyrgyzstan',
- 'LA': 'Lao People\'s Democratic Republic',
- 'LV': 'Latvia',
- 'LB': 'Lebanon',
- 'LS': 'Lesotho',
- 'LR': 'Liberia',
- 'LY': 'Libyan Arab Jamahiriya',
- 'LI': 'Liechtenstein',
- 'LT': 'Lithuania',
- 'LU': 'Luxembourg',
- 'MO': 'Macao',
- 'MK': 'Macedonia',
- 'MG': 'Madagascar',
- 'MW': 'Malawi',
- 'MY': 'Malaysia',
- 'MV': 'Maldives',
- 'ML': 'Mali',
- 'MT': 'Malta',
- 'MH': 'Marshall Islands',
- 'MQ': 'Martinique',
- 'MR': 'Mauritania',
- 'MU': 'Mauritius',
- 'YT': 'Mayotte',
- 'MX': 'Mexico',
- 'FM': 'Micronesia, Federated States Of',
- 'MD': 'Moldova',
- 'MC': 'Monaco',
- 'MN': 'Mongolia',
- 'ME': 'Montenegro',
- 'MS': 'Montserrat',
- 'MA': 'Morocco',
- 'MZ': 'Mozambique',
- 'MM': 'Myanmar',
- 'NA': 'Namibia',
- 'NR': 'Nauru',
- 'NP': 'Nepal',
- 'NL': 'Netherlands',
- 'AN': 'Netherlands Antilles',
- 'NC': 'New Caledonia',
- 'NZ': 'New Zealand',
- 'NI': 'Nicaragua',
- 'NE': 'Niger',
- 'NG': 'Nigeria',
- 'NU': 'Niue',
- 'NF': 'Norfolk Island',
- 'MP': 'Northern Mariana Islands',
- 'NO': 'Norway',
- 'OM': 'Oman',
- 'PK': 'Pakistan',
- 'PW': 'Palau',
- 'PS': 'Palestinian Territory, Occupied',
- 'PA': 'Panama',
- 'PG': 'Papua New Guinea',
- 'PY': 'Paraguay',
- 'PE': 'Peru',
- 'PH': 'Philippines',
- 'PN': 'Pitcairn',
- 'PL': 'Poland',
- 'PT': 'Portugal',
- 'PR': 'Puerto Rico',
- 'QA': 'Qatar',
- 'RE': 'Reunion',
- 'RO': 'Romania',
- 'RU': 'Russian Federation',
- 'RW': 'Rwanda',
- 'BL': 'Saint Barthelemy',
- 'SH': 'Saint Helena',
- 'KN': 'Saint Kitts And Nevis',
- 'LC': 'Saint Lucia',
- 'MF': 'Saint Martin',
- 'PM': 'Saint Pierre And Miquelon',
- 'VC': 'Saint Vincent And Grenadines',
- 'WS': 'Samoa',
- 'SM': 'San Marino',
- 'ST': 'Sao Tome And Principe',
- 'SA': 'Saudi Arabia',
- 'SN': 'Senegal',
- 'RS': 'Serbia',
- 'SC': 'Seychelles',
- 'SL': 'Sierra Leone',
- 'SG': 'Singapore',
- 'SK': 'Slovakia',
- 'SI': 'Slovenia',
- 'SB': 'Solomon Islands',
- 'SO': 'Somalia',
- 'ZA': 'South Africa',
- 'GS': 'South Georgia And Sandwich Isl.',
- 'ES': 'Spain',
- 'LK': 'Sri Lanka',
- 'SD': 'Sudan',
- 'SR': 'Suriname',
- 'SJ': 'Svalbard And Jan Mayen',
- 'SZ': 'Swaziland',
- 'SE': 'Sweden',
- 'CH': 'Switzerland',
- 'SY': 'Syrian Arab Republic',
- 'TW': 'Taiwan',
- 'TJ': 'Tajikistan',
- 'TZ': 'Tanzania',
- 'TH': 'Thailand',
- 'TL': 'Timor-Leste',
- 'TG': 'Togo',
- 'TK': 'Tokelau',
- 'TO': 'Tonga',
- 'TT': 'Trinidad And Tobago',
- 'TN': 'Tunisia',
- 'TR': 'Turkey',
- 'TM': 'Turkmenistan',
- 'TC': 'Turks And Caicos Islands',
- 'TV': 'Tuvalu',
- 'UG': 'Uganda',
- 'UA': 'Ukraine',
- 'AE': 'United Arab Emirates',
- 'GB': 'United Kingdom',
- 'US': 'United States',
- 'UM': 'United States Outlying Islands',
- 'UY': 'Uruguay',
- 'UZ': 'Uzbekistan',
- 'VU': 'Vanuatu',
- 'VE': 'Venezuela',
- 'VN': 'Viet Nam',
- 'VG': 'Virgin Islands, British',
- 'VI': 'Virgin Islands, U.S.',
- 'WF': 'Wallis And Futuna',
- 'EH': 'Western Sahara',
- 'YE': 'Yemen',
- 'ZM': 'Zambia',
- 'ZW': 'Zimbabwe'
- });
-})();
+ .controller('MapController', MapController);
+
+ MapController.$inject = ['$scope', '$state', '$stateParams', '$timeout', 'device',
+ '$mdDialog', 'leafletData', 'alert',
+ 'Marker', 'tag', 'animation', '$q'];
+ function MapController($scope, $state, $stateParams, $timeout, device,
+ $mdDialog, leafletData, alert, Marker, tag, animation, $q) {
+ var vm = this;
+ var updateType;
+ var focusedMarkerID;
+
+ vm.markers = [];
+
+ var retinaSuffix = isRetina() ? '512' : '256';
+ var retinaLegacySuffix = isRetina() ? '@2x' : '';
+
+ var mapBoxToken = 'pk.eyJ1IjoidG9tYXNkaWV6IiwiYSI6ImRTd01HSGsifQ.loQdtLNQ8GJkJl2LUzzxVg';
+
+ vm.layers = {
+ baselayers: {
+ osm: {
+ name: 'OpenStreetMap',
+ type: 'xyz',
+ url: 'https://api.mapbox.com/styles/v1/mapbox/streets-v10/tiles/' + retinaSuffix + '/{z}/{x}/{y}?access_token=' + mapBoxToken
+ },
+ legacy: {
+ name: 'Legacy',
+ type: 'xyz',
+ url: 'https://api.tiles.mapbox.com/v4/mapbox.streets-basic/{z}/{x}/{y}'+ retinaLegacySuffix +'.png' + '?access_token=' + mapBoxToken
+ },
+ sat: {
+ name: 'Satellite',
+ type: 'xyz',
+ url: 'https://api.mapbox.com/styles/v1/mapbox/satellite-streets-v10/tiles/' + retinaSuffix + '/{z}/{x}/{y}?access_token=' + mapBoxToken
+ }
+ },
+ overlays: {
+ devices: {
+ name: 'Devices',
+ type: 'markercluster',
+ visible: true,
+ layerOptions: {
+ showCoverageOnHover: false
+ }
+ }
+ }
+ };
+
+ vm.center = {
+ lat: $stateParams.lat ? parseInt($stateParams.lat, 10) : 13.14950321154457,
+ lng: $stateParams.lng ? parseInt($stateParams.lng, 10) : -1.58203125,
+ zoom: $stateParams.zoom ? parseInt($stateParams.zoom, 10) : 2
+ };
+
+ vm.defaults = {
+ dragging: true,
+ touchZoom: true,
+ scrollWheelZoom: true,
+ doubleClickZoom: true,
+ minZoom:2,
+ worldCopyJump: true
+ };
+
+ vm.events = {
+ map: {
+ enable: ['dragend', 'zoomend', 'moveend', 'popupopen', 'popupclose',
+ 'mousedown', 'dblclick', 'click', 'touchstart', 'mouseup'],
+ logic: 'broadcast'
+ }
+ };
+
+ $scope.$on('leafletDirectiveMarker.click', function(event, data) {
+ var id = undefined;
+ var currentMarker = vm.markers[data.modelName];
+
+ if(currentMarker) {
+ id = currentMarker.myData.id;
+ }
+
+ vm.deviceLoading = true;
+ vm.center.lat = data.leafletEvent.latlng.lat;
+ vm.center.lng = data.leafletEvent.latlng.lng;
+
+ if(id === parseInt($state.params.id)) {
+ $timeout(function() {
+ vm.deviceLoading = false;
+ });
+ return;
+ }
+
+ updateType = 'map';
+
+ if ($state.$current.name === 'embbed') { return; }
+ $state.go('layout.home.kit', {id: id});
+
+ // angular.element('section.map').scope().$broadcast('resizeMapHeight');
+ });
-(function() {
- 'use strict';
- /**
- * Unused directive. Double-check before removing.
- *
- */
- angular.module('app.components')
- .directive('slide', slide)
- .directive('slideMenu', slideMenu);
+ $scope.$on('leafletDirectiveMarker.popupclose', function() {
+ if(focusedMarkerID) {
+ var marker = vm.markers[focusedMarkerID];
+ if(marker) {
+ vm.markers[focusedMarkerID].focus = false;
+ }
+ }
+ });
- function slideMenu() {
- return {
- controller: controller,
- link: link
+ vm.readyForDevice = {
+ device: false,
+ map: false
};
- function link(scope, element) {
- scope.element = element;
- }
+ $scope.$on('deviceLoaded', function(event, data) {
+ vm.readyForDevice.device = data;
+ });
- function controller($scope) {
- $scope.slidePosition = 0;
- $scope.slideSize = 20;
+ $scope.$watch('vm.readyForDevice', function() {
+ if (vm.readyForDevice.device && vm.readyForDevice.map) {
+ zoomDeviceAndPopUp(vm.readyForDevice.device);
+ }
+ }, true);
- this.getTimesSlided = function() {
- return $scope.slideSize;
- };
- this.getPosition = function() {
- return $scope.slidePosition * $scope.slideSize;
- };
- this.decrementPosition = function() {
- $scope.slidePosition -= 1;
- };
- this.incrementPosition = function() {
- $scope.slidePosition += 1;
- };
- this.scrollIsValid = function(direction) {
- var scrollPosition = $scope.element.scrollLeft();
- console.log('scrollpos', scrollPosition);
- if(direction === 'left') {
- return scrollPosition > 0 && $scope.slidePosition >= 0;
- } else if(direction === 'right') {
- return scrollPosition < 300;
- }
- };
+ $scope.$on('goToLocation', function(event, data) {
+ goToLocation(data);
+ });
+
+ vm.filters = ['indoor', 'outdoor', 'online', 'offline'];
+
+ vm.openFilterPopup = openFilterPopup;
+ vm.openTagPopup = openTagPopup;
+ vm.removeFilter = removeFilter;
+ vm.removeTag = removeTag;
+ vm.selectedTags = tag.getSelectedTags();
+ vm.selectedFilters = ['indoor', 'outdoor', 'online', 'offline', 'new'];
+
+ vm.checkAllFiltersSelected = checkAllFiltersSelected;
+
+ initialize();
+
+ /////////////////////
+
+ function initialize() {
+
+ vm.readyForDevice.map = false;
+
+ $q.all([device.getAllDevices($stateParams.reloadMap)])
+ .then(function(data){
+
+ data = data[0];
+
+ vm.markers = _.chain(data)
+ .map(function(device) {
+ return new Marker(device);
+ })
+ .filter(function(marker) {
+ return !!marker.lng && !!marker.lat;
+ })
+ .tap(function(data) {
+ device.setWorldMarkers(data);
+ })
+ .value();
+
+ var markersByIndex = _.keyBy(vm.markers, function(marker) {
+ return marker.myData.id;
+ });
+
+ if($state.params.id && markersByIndex[parseInt($state.params.id)]){
+ focusedMarkerID = markersByIndex[parseInt($state.params.id)]
+ .myData.id;
+ vm.readyForDevice.map = true;
+ } else {
+ updateMarkers();
+ vm.readyForDevice.map = true;
+ }
+
+ });
}
- }
- slide.$inject = [];
- function slide() {
- return {
- link: link,
- require: '^slide-menu',
- restrict: 'A',
- scope: {
- direction: '@'
+ function zoomDeviceAndPopUp(data){
+
+ if(updateType === 'map') {
+ vm.deviceLoading = false;
+ updateType = undefined;
+ return;
+ } else {
+ vm.deviceLoading = true;
}
- };
- function link(scope, element, attr, slideMenuCtrl) {
- //select first sensor container
- var sensorsContainer = angular.element('.sensors_container');
+ leafletData.getMarkers()
+ .then(function(markers) {
+ var currentMarker = _.find(markers, function(marker) {
+ return data.id === marker.options.myData.id;
+ });
- element.on('click', function() {
+ var id = data.id;
- if(slideMenuCtrl.scrollIsValid('left') && attr.direction === 'left') {
- slideMenuCtrl.decrementPosition();
- sensorsContainer.scrollLeft(slideMenuCtrl.getPosition());
- console.log(slideMenuCtrl.getPosition());
- } else if(slideMenuCtrl.scrollIsValid('right') && attr.direction === 'right') {
- slideMenuCtrl.incrementPosition();
- sensorsContainer.scrollLeft(slideMenuCtrl.getPosition());
- console.log(slideMenuCtrl.getPosition());
- }
- });
- }
- }
-})();
+ leafletData.getLayers()
+ .then(function(layers) {
+ if(currentMarker){
+ layers.overlays.devices.zoomToShowLayer(currentMarker,
+ function() {
+ var selectedMarker = currentMarker;
+ if(selectedMarker) {
+ // Ensures the marker is not just zoomed but the marker is centered to improve UX
+ // The $timeout can be replaced by an event but tests didn't show good results
+ $timeout(function() {
+ vm.center.lat = selectedMarker.options.lat;
+ vm.center.lng = selectedMarker.options.lng;
+ selectedMarker.openPopup();
+ vm.deviceLoading = false;
+ }, 1000);
+ }
+ });
+ } else {
+ leafletData.getMap().then(function(map){
+ map.closePopup();
+ });
+ }
+ });
+ });
-(function() {
- 'use strict';
+ }
- angular.module('app.components')
- .directive('showPopupInfo', showPopupInfo);
+ function checkAllFiltersSelected() {
+ var allFiltersSelected = _.every(vm.filters, function(filterValue) {
+ return _.includes(vm.selectedFilters, filterValue);
+ });
+ return allFiltersSelected;
+ }
- /**
- * Used to show/hide explanation of sensor value at kit dashboard
- *
- */
- showPopupInfo.$inject = [];
- function showPopupInfo() {
- return {
- link: link
- };
+ function openFilterPopup() {
+ $mdDialog.show({
+ hasBackdrop: true,
+ controller: 'MapFilterModalController',
+ controllerAs: 'vm',
+ templateUrl: 'app/components/map/mapFilterModal.html',
+ clickOutsideToClose: true,
+ locals: {
+ selectedFilters: vm.selectedFilters
+ }
+ })
+ .then(function(selectedFilters) {
+ updateType = 'map';
+ vm.selectedFilters = selectedFilters;
+ updateMapFilters();
+ });
+ }
- //////
+ function openTagPopup() {
+ $mdDialog.show({
+ hasBackdrop: true,
+ controller: 'MapTagModalController',
+ controllerAs: 'vm',
+ templateUrl: 'app/components/map/mapTagModal.html',
+ //targetEvent: ev,
+ clickOutsideToClose: true,
+ locals: {
+ selectedTags: vm.selectedTags
+ }
+ })
+ .then(function(selectedTags) {
+ if (selectedTags && selectedTags.length > 0) {
+ updateType = 'map';
+ tag.setSelectedTags(_.map(selectedTags, 'name'));
+ vm.selectedTags = tag.getSelectedTags();
+ reloadWithTags();
+ } else if (selectedTags === null) {
+ reloadNoTags();
+ }
+ });
+ }
+ function updateMapFilters(){
+ vm.selectedTags = tag.getSelectedTags();
+ checkAllFiltersSelected();
+ updateMarkers();
+ }
- function link(scope, elem) {
- elem.on('mouseenter', function() {
- angular.element('.sensor_data_description').css('display', 'inline-block');
+ function removeFilter(filterName) {
+ vm.selectedFilters = _.filter(vm.selectedFilters, function(el){
+ return el !== filterName;
});
- elem.on('mouseleave', function() {
- angular.element('.sensor_data_description').css('display', 'none');
+ if(vm.selectedFilters.length === 0){
+ vm.selectedFilters = vm.filters;
+ }
+ updateMarkers();
+ }
+
+ function filterMarkersByLabel(tmpMarkers) {
+ return tmpMarkers.filter(function(marker) {
+ var labels = marker.myData.labels;
+ if (labels.length === 0 && vm.selectedFilters.length !== 0){
+ return false;
+ }
+ return _.every(labels, function(label) {
+ return _.includes(vm.selectedFilters, label);
+ });
});
}
- }
-})();
-(function() {
- 'use strict';
+ function updateMarkers() {
+ $timeout(function() {
+ $scope.$apply(function() {
+ var allMarkers = device.getWorldMarkers();
- angular.module('app.components')
- .directive('showPopup', showPopup);
+ var updatedMarkers = allMarkers;
- /**
- * Used on kit dashboard to open full sensor description
- */
+ updatedMarkers = tag.filterMarkersByTag(updatedMarkers);
+ updatedMarkers = filterMarkersByLabel(updatedMarkers);
+ vm.markers = updatedMarkers;
- showPopup.$inject = [];
- function showPopup() {
- return {
- link: link
- };
+ animation.mapStateLoaded();
- /////
+ vm.deviceLoading = false;
- function link(scope, element) {
- element.on('click', function() {
- var text = angular.element('.sensor_description_preview').text();
- if(text.length < 140) {
- return;
- }
- angular.element('.sensor_description_preview').hide();
- angular.element('.sensor_description_full').show();
+ zoomOnMarkers();
+ });
});
}
- }
-})();
-(function() {
- 'use strict';
+ function getZoomLevel(data) {
+ // data.layer is an array of strings like ["establishment", "point_of_interest"]
+ var zoom = 18;
- angular.module('app.components')
- .directive('moveFilters', moveFilters);
+ if(data.layer && data.layer[0]) {
+ switch(data.layer[0]) {
+ case 'point_of_interest':
+ zoom = 18;
+ break;
+ case 'address':
+ zoom = 18;
+ break;
+ case "establishment":
+ zoom = 15;
+ break;
+ case 'neighbourhood':
+ zoom = 13;
+ break;
+ case 'locality':
+ zoom = 13;
+ break;
+ case 'localadmin':
+ zoom = 9;
+ break;
+ case 'county':
+ zoom = 9;
+ break;
+ case 'region':
+ zoom = 8;
+ break;
+ case 'country':
+ zoom = 7;
+ break;
+ case 'coarse':
+ zoom = 7;
+ break;
+ }
+ }
- /**
- * Moves map filters when scrolling
- *
- */
- moveFilters.$inject = ['$window', '$timeout'];
- function moveFilters($window, $timeout) {
- return {
- link: link
- };
+ return zoom;
+ }
- function link() {
- var chartHeight;
- $timeout(function() {
- chartHeight = angular.element('.kit_chart').height();
- }, 1000);
+ function isRetina(){
+ return ((window.matchMedia &&
+ (window.matchMedia('only screen and (min-resolution: 192dpi), ' +
+ 'only screen and (min-resolution: 2dppx), only screen and ' +
+ '(min-resolution: 75.6dpcm)').matches ||
+ window.matchMedia('only screen and (-webkit-min-device-pixel-ra' +
+ 'tio: 2), only screen and (-o-min-device-pixel-ratio: 2/1), only' +
+ ' screen and (min--moz-device-pixel-ratio: 2), only screen and ' +
+ '(min-device-pixel-ratio: 2)').matches)) ||
+ (window.devicePixelRatio && window.devicePixelRatio >= 2)) &&
+ /(iPad|iPhone|iPod|Apple)/g.test(navigator.userAgent);
+ }
- /*
- angular.element($window).on('scroll', function() {
- var windowPosition = document.body.scrollTop;
- if(chartHeight > windowPosition) {
- elem.css('bottom', 12 + windowPosition + 'px');
- }
+ function goToLocation(data){
+ // This ensures the action runs after the event is registered
+ $timeout(function() {
+ vm.center.lat = data.lat;
+ vm.center.lng = data.lng;
+ vm.center.zoom = getZoomLevel(data);
});
- */
}
- }
-})();
-(function() {
- 'use strict';
-
- angular.module('app.components')
- .factory('layout', layout);
+ function removeTag(tagName){
+ tag.setSelectedTags(_.filter(vm.selectedTags, function(el){
+ return el !== tagName;
+ }));
+ vm.selectedTags = tag.getSelectedTags();
- function layout() {
+ if(vm.selectedTags.length === 0){
+ reloadNoTags();
+ } else {
+ reloadWithTags();
+ }
- var kitHeight;
+ }
- var service = {
- setKit: setKit,
- getKit: getKit
- };
- return service;
+ function zoomOnMarkers(){
+ $timeout(function() {
+ if(vm.markers && vm.markers.length > 0) {
+ leafletData.getMap().then(function(map){
+ var bounds = L.latLngBounds(vm.markers);
+ map.fitBounds(bounds);
+ });
+ } else {
+ alert.error('No markers found with those filters', 5000);
+ }
+ });
+ }
- function setKit(height) {
- kitHeight = height;
+ function reloadWithTags(){
+ $state.transitionTo('layout.home.tags', {tags: vm.selectedTags}, {reload: true});
}
- function getKit() {
- return kitHeight;
+ function reloadNoTags(){
+ $state.transitionTo('layout.home.kit');
}
+
}
+
})();
(function() {
'use strict';
angular.module('app.components')
- .directive('horizontalScroll', horizontalScroll);
-
- /**
- * Used to highlight and unhighlight buttons on the kit dashboard when scrolling horizontally
- *
- */
- horizontalScroll.$inject = ['$window', '$timeout'];
- function horizontalScroll($window, $timeout) {
- return {
- link: link,
- restrict: 'A'
- };
-
- ///////////////////
-
-
- function link(scope, element) {
-
- element.on('scroll', function() {
- // horizontal scroll position
- var position = angular.element(this).scrollLeft();
- // real width of element
- var scrollWidth = this.scrollWidth;
- // visible width of element
- var width = angular.element(this).width();
+ .controller('LoginModalController', LoginModalController);
- // if you cannot scroll, unhighlight both
- if(scrollWidth === width) {
- angular.element('.button_scroll_left').css('opacity', '0.5');
- angular.element('.button_scroll_right').css('opacity', '0.5');
- }
- // if scroll is in the middle, highlight both
- if(scrollWidth - width > 2) {
- angular.element('.button_scroll_left').css('opacity', '1');
- angular.element('.button_scroll_right').css('opacity', '1');
- }
- // if scroll is at the far right, unhighligh right button
- if(scrollWidth - width - position <= 2) {
- angular.element('.button_scroll_right').css('opacity', '0.5');
- return;
- }
- // if scroll is at the far left, unhighligh left button
- if(position === 0) {
- angular.element('.button_scroll_left').css('opacity', '0.5');
- return;
- }
+ LoginModalController.$inject = ['$scope', '$mdDialog', 'auth', 'animation'];
+ function LoginModalController($scope, $mdDialog, auth, animation) {
+ const vm = this;
+ $scope.answer = function(answer) {
+ $scope.waitingFromServer = true;
+ auth.login(answer)
+ .then(function(data) {
+ /*jshint camelcase: false */
+ var token = data.access_token;
+ auth.saveToken(token);
+ $mdDialog.hide();
+ })
+ .catch(function(err) {
+ vm.errors = err.data;
+ })
+ .finally(function() {
+ $scope.waitingFromServer = false;
+ });
+ };
+ $scope.hide = function() {
+ $mdDialog.hide();
+ };
+ $scope.cancel = function() {
+ $mdDialog.hide();
+ };
- //set opacity back to normal otherwise
- angular.element('.button_scroll_left').css('opacity', '1');
- angular.element('.button_scroll_right').css('opacity', '1');
- });
+ $scope.openSignup = function() {
+ animation.showSignup();
+ $mdDialog.hide();
+ };
- $timeout(function() {
- element.trigger('scroll');
- });
+ $scope.openPasswordRecovery = function() {
+ $mdDialog.show({
+ hasBackdrop: true,
+ controller: 'PasswordRecoveryModalController',
+ templateUrl: 'app/components/passwordRecovery/passwordRecoveryModal.html',
+ clickOutsideToClose: true
+ });
- angular.element($window).on('resize', function() {
- $timeout(function() {
- element.trigger('scroll');
- }, 1000);
- });
+ $mdDialog.hide();
+ };
}
- }
})();
(function() {
'use strict';
- angular.module('app.components')
- .directive('hidePopup', hidePopup);
+ angular.module('app.components')
+ .directive('login', login);
- /**
- * Used on kit dashboard to hide popup with full sensor description
- *
- */
-
- hidePopup.$inject = [];
- function hidePopup() {
+ function login() {
return {
- link: link
+ scope: {
+ show: '='
+ },
+ restrict: 'A',
+ controller: 'LoginController',
+ controllerAs: 'vm',
+ templateUrl: 'app/components/login/login.html'
};
-
- /////////////
-
- function link(scope, elem) {
- elem.on('mouseleave', function() {
- angular.element('.sensor_description_preview').show();
- angular.element('.sensor_description_full').hide();
- });
- }
}
})();
@@ -6316,1095 +6327,1084 @@ function cookiesLaw($cookies) {
'use strict';
angular.module('app.components')
- .directive('disableScroll', disableScroll);
+ .controller('LoginController', LoginController);
- disableScroll.$inject = ['$timeout'];
- function disableScroll($timeout) {
- return {
- // link: {
- // pre: link
- // },
- compile: link,
- restrict: 'A',
- priority: 100000
- };
+ LoginController.$inject = ['$scope', '$mdDialog'];
+ function LoginController($scope, $mdDialog) {
+ $scope.showLogin = showLogin;
- //////////////////////
+ $scope.$on('showLogin', function() {
+ showLogin();
+ });
- function link(elem) {
- console.log('i', elem);
- // var select = elem.find('md-select');
- // angular.element(select).on('click', function() {
- elem.on('click', function() {
- console.log('e');
- angular.element(document.body).css('overflow', 'hidden');
- $timeout(function() {
- angular.element(document.body).css('overflow', 'initial');
- });
- });
- }
+ ////////////////
+
+ function showLogin() {
+ $mdDialog.show({
+ hasBackdrop: true,
+ fullscreen: true,
+ controller: 'LoginModalController',
+ controllerAs: 'vm',
+ templateUrl: 'app/components/login/loginModal.html',
+ clickOutsideToClose: true
+ });
}
+
+ }
})();
(function() {
'use strict';
angular.module('app.components')
- .factory('animation', animation);
-
- /**
- * Used to emit events from rootscope.
- *
- * This events are then listened by $scope on controllers and directives that care about that particular event
- */
+ .controller('LayoutController', LayoutController);
- animation.$inject = ['$rootScope'];
- function animation($rootScope) {
+ LayoutController.$inject = ['$mdSidenav','$mdDialog', '$location', '$state', '$scope', '$transitions', 'auth', 'animation', '$timeout', 'DROPDOWN_OPTIONS_COMMUNITY', 'DROPDOWN_OPTIONS_USER'];
+ function LayoutController($mdSidenav, $mdDialog, $location, $state, $scope, $transitions, auth, animation, $timeout, DROPDOWN_OPTIONS_COMMUNITY, DROPDOWN_OPTIONS_USER) {
+ var vm = this;
- var service = {
- blur: blur,
- unblur: unblur,
- removeNav: removeNav,
- addNav: addNav,
- showChartSpinner: showChartSpinner,
- hideChartSpinner: hideChartSpinner,
- deviceLoaded: deviceLoaded,
- showPasswordRecovery: showPasswordRecovery,
- showLogin: showLogin,
- showSignup: showSignup,
- showPasswordReset: showPasswordReset,
- hideAlert: hideAlert,
- viewLoading: viewLoading,
- viewLoaded: viewLoaded,
- deviceWithoutData: deviceWithoutData,
- deviceIsPrivate: deviceIsPrivate,
- goToLocation: goToLocation,
- mapStateLoading: mapStateLoading,
- mapStateLoaded: mapStateLoaded
- };
- return service;
+ vm.navRightLayout = 'space-around center';
- //////////////
+ $scope.toggleRight = buildToggler('right');
- function blur() {
- $rootScope.$broadcast('blur');
- }
- function unblur() {
- $rootScope.$broadcast('unblur');
- }
- function removeNav() {
- $rootScope.$broadcast('removeNav');
- }
- function addNav() {
- $rootScope.$broadcast('addNav');
- }
- function showChartSpinner() {
- $rootScope.$broadcast('showChartSpinner');
- }
- function hideChartSpinner() {
- $rootScope.$broadcast('hideChartSpinner');
- }
- function deviceLoaded(data) {
- $rootScope.$broadcast('deviceLoaded', data);
- }
- function showPasswordRecovery() {
- $rootScope.$broadcast('showPasswordRecovery');
- }
- function showLogin() {
- $rootScope.$broadcast('showLogin');
- }
- function showSignup() {
- $rootScope.$broadcast('showSignup');
- }
- function showPasswordReset() {
- $rootScope.$broadcast('showPasswordReset');
- }
- function hideAlert() {
- $rootScope.$broadcast('hideAlert');
- }
- function viewLoading() {
- $rootScope.$broadcast('viewLoading');
- }
- function viewLoaded() {
- $rootScope.$broadcast('viewLoaded');
- }
- function deviceWithoutData(data) {
- $rootScope.$broadcast('deviceWithoutData', data);
- }
- function deviceIsPrivate(data) {
- $rootScope.$broadcast('deviceIsPrivate', data);
- }
- function goToLocation(data) {
- $rootScope.$broadcast('goToLocation', data);
- }
- function mapStateLoading() {
- $rootScope.$broadcast('mapStateLoading');
- }
- function mapStateLoaded() {
- $rootScope.$broadcast('mapStateLoaded');
+ function buildToggler(componentId) {
+ return function() {
+ $mdSidenav(componentId).toggle();
+ };
}
- }
-})();
-
-(function() {
- 'use strict';
- /**
- * TODO: Improvement These directives can be split up each one in a different file
- */
+ // listen for any login event so that the navbar can be updated
+ $scope.$on('loggedIn', function(ev, options) {
+ // if(options && options.time === 'appLoad') {
+ // $scope.$apply(function() {
+ // vm.isLoggedin = true;
+ // vm.isShown = true;
+ // angular.element('.nav_right .wrap-dd-menu').css('display', 'initial');
+ // vm.currentUser = auth.getCurrentUser().data;
+ // vm.dropdownOptions[0].text = 'Hello, ' + vm.currentUser.username;
+ // vm.navRightLayout = 'end center';
+ // });
+ // } else {
+ // vm.isLoggedin = true;
+ // vm.isShown = true;
+ // angular.element('.nav_right .wrap-dd-menu').css('display', 'initial');
+ // vm.currentUser = auth.getCurrentUser().data;
+ // vm.dropdownOptions[0].text = 'Hello, ' + vm.currentUser.username;
+ // vm.navRightLayout = 'end center';
+ // }
- angular.module('app.components')
- .directive('moveDown', moveDown)
- .directive('stick', stick)
- .directive('blur', blur)
- .directive('focus', focus)
- .directive('changeMapHeight', changeMapHeight)
- .directive('changeContentMargin', changeContentMargin)
- .directive('focusInput', focusInput);
+ vm.isLoggedin = true;
+ vm.isShown = true;
+ angular.element('.nav_right .wrap-dd-menu').css('display', 'initial');
+ vm.currentUser = auth.getCurrentUser().data;
+ vm.dropdownOptions[0].text = 'Hi, ' + vm.currentUser.username + '!';
+ vm.navRightLayout = 'end center';
+ if(!$scope.$$phase) {
+ $scope.$digest();
+ }
+ });
- /**
- * It moves down kit section to ease the transition after the kit menu is sticked to the top
- *
- */
- moveDown.$inject = [];
- function moveDown() {
+ // listen for logout events so that the navbar can be updated
+ $scope.$on('loggedOut', function() {
+ vm.isLoggedIn = false;
+ vm.isShown = true;
+ angular.element('navbar .wrap-dd-menu').css('display', 'none');
+ vm.navRightLayout = 'space-around center';
+ });
- function link(scope, element) {
- scope.$watch('moveDown', function(isTrue) {
- if(isTrue) {
- element.addClass('move_down');
- } else {
- element.removeClass('move_down');
- }
- });
- }
- return {
- link: link,
- scope: false,
- restrict: 'A'
- };
- }
+ vm.isShown = true;
+ vm.isLoggedin = false;
+ vm.logout = logout;
- /**
- * It sticks kit menu when kit menu touchs navbar on scrolling
- *
- */
- stick.$inject = ['$window', '$timeout'];
- function stick($window, $timeout) {
- function link(scope, element) {
- var elementPosition = element[0].offsetTop;
- //var elementHeight = element[0].offsetHeight;
- var navbarHeight = angular.element('.stickNav').height();
+ vm.dropdownOptions = DROPDOWN_OPTIONS_USER;
+ vm.dropdownSelected = undefined;
- $timeout(function() {
- elementPosition = element[0].offsetTop;
- //var elementHeight = element[0].offsetHeight;
- navbarHeight = angular.element('.stickNav').height();
- }, 1000);
+ vm.dropdownOptionsCommunity = DROPDOWN_OPTIONS_COMMUNITY;
+ vm.dropdownSelectedCommunity = undefined;
+ $scope.$on('removeNav', function() {
+ vm.isShown = false;
+ });
- angular.element($window).on('scroll', function() {
- var windowPosition = document.body.scrollTop;
+ $scope.$on('addNav', function() {
+ vm.isShown = true;
+ });
- //sticking menu and moving up/down
- if(windowPosition + navbarHeight >= elementPosition) {
- element.addClass('stickMenu');
- scope.$apply(function() {
- scope.moveDown = true;
- });
- } else {
- element.removeClass('stickMenu');
- scope.$apply(function() {
- scope.moveDown = false;
- });
+ initialize();
+
+ //////////////////
+
+ function initialize() {
+ $timeout(function() {
+ var hash = $location.search();
+ if(hash.signup) {
+ animation.showSignup();
+ } else if(hash.login) {
+ animation.showLogin();
+ } else if(hash.passwordRecovery) {
+ animation.showPasswordRecovery();
}
- });
+ }, 1000);
}
- return {
- link: link,
- scope: false,
- restrict: 'A'
- };
+ function logout() {
+ auth.logout();
+ vm.isLoggedin = false;
+ }
}
+})();
- /**
- * Unused directive. Double-check is not being used before removing it
- *
- */
+(function() {
+ 'use strict';
- function blur() {
+ angular.module('app.components')
+ .controller('LandingController', LandingController);
- function link(scope, element) {
+ LandingController.$inject = ['$timeout', 'animation', '$mdDialog', '$location', '$anchorScroll'];
- scope.$on('blur', function() {
- element.addClass('blur');
- });
+ function LandingController($timeout, animation, $mdDialog, $location, $anchorScroll) {
+ var vm = this;
- scope.$on('unblur', function() {
- element.removeClass('blur');
- });
- }
+ vm.showStore = showStore;
+ vm.goToHash = goToHash;
- return {
- link: link,
- scope: false,
- restrict: 'A'
- };
- }
+ ///////////////////////
- /**
- * Used to remove nav and unable scrolling when searching
- *
- */
- focus.$inject = ['animation'];
- function focus(animation) {
- function link(scope, element) {
- element.on('focusin', function() {
- animation.removeNav();
- });
+ initialize();
- element.on('focusout', function() {
- animation.addNav();
- });
+ //////////////////
- var searchInput = element.find('input');
- searchInput.on('blur', function() {
- //enable scrolling on body when search input is not active
- angular.element(document.body).css('overflow', 'auto');
- });
+ function initialize() {
+ $timeout(function() {
+ animation.viewLoaded();
+ if($location.hash()) {
+ $anchorScroll();
+ }
+ }, 500);
+ }
- searchInput.on('focus', function() {
- angular.element(document.body).css('overflow', 'hidden');
- });
- }
+ function goToHash(hash){
+ $location.hash(hash);
+ $anchorScroll();
+ }
- return {
- link: link
- };
+ function showStore() {
+ $mdDialog.show({
+ hasBackdrop: true,
+ controller: 'StoreModalController',
+ templateUrl: 'app/components/store/storeModal.html',
+ clickOutsideToClose: true
+ });
}
+ }
+})();
- /**
- * Changes map section based on screen size
- *
- */
- changeMapHeight.$inject = ['$document', 'layout', '$timeout'];
- function changeMapHeight($document, layout, $timeout) {
- function link(scope, element) {
+(function(){
+ 'use strict';
+ angular.module('app.components')
+ .directive('kitList',kitList);
- var screenHeight = $document[0].body.clientHeight;
- var navbarHeight = angular.element('.stickNav').height();
+ function kitList(){
+ return{
+ restrict:'E',
+ scope:{
+ devices:'=devices',
+ actions: '=actions'
+ },
+ controllerAs:'vm',
+ templateUrl:'app/components/kitList/kitList.html'
+ };
+ }
+})();
- // var overviewHeight = angular.element('.kit_overview').height();
- // var menuHeight = angular.element('.kit_menu').height();
- // var chartHeight = angular.element('.kit_chart').height();
+(function() {
+ 'use strict';
- function resizeMap(){
- $timeout(function() {
- var overviewHeight = angular.element('.over_map').height();
+ angular.module('app.components')
+ .controller('HomeController', HomeController);
- var objectsHeight = navbarHeight + overviewHeight;
- var objectsHeightPercentage = parseInt((objectsHeight * 100) / screenHeight);
- var mapHeightPercentage = 100 - objectsHeightPercentage;
+ function HomeController() {
+ }
+})();
+(function (){
+ 'use strict';
- element.css('height', mapHeightPercentage + '%');
+ angular.module('app.components')
+ .controller('DownloadModalController', DownloadModalController);
- var aboveTheFoldHeight = screenHeight - overviewHeight;
- angular
- .element('section[change-content-margin]')
- .css('margin-top', aboveTheFoldHeight + 'px');
- });
- }
+ DownloadModalController.$inject = ['thisDevice', 'device', '$mdDialog'];
- resizeMap();
+ function DownloadModalController(thisDevice, device, $mdDialog) {
+ var vm = this;
- scope.element = element;
+ vm.device = thisDevice;
+ vm.download = download;
+ vm.cancel = cancel;
- scope.$on('resizeMapHeight',function(){
- resizeMap();
- });
+ ////////////////////////////
- }
+ function download(){
+ device.mailReadings(vm.device)
+ .then(function (){
+ $mdDialog.hide();
+ }).catch(function(err){
+ $mdDialog.cancel(err);
+ });
+ }
- return {
- link: link,
- scope: true,
- restrict: 'A'
- };
- }
+ function cancel(){
+ $mdDialog.cancel();
+ }
+ }
- /**
- * Changes margin on kit section based on above-the-fold space left after map section is resize
- */
+})();
- changeContentMargin.$inject = ['layout', '$timeout', '$document'];
- function changeContentMargin(layout, $timeout, $document) {
- function link(scope, element) {
- var screenHeight = $document[0].body.clientHeight;
+(function(){
+'use strict';
- var overviewHeight = angular.element('.over_map').height();
+angular.module('app.components')
+ .directive('cookiesLaw', cookiesLaw);
+
+
+cookiesLaw.$inject = ['$cookies'];
+
+function cookiesLaw($cookies) {
+ return {
+ template:
+ '' +
+ 'This site uses cookies to offer you a better experience. ' +
+ '
Accept or' +
+ '
Learn More. ' +
+ '
',
+ controller: function($scope) {
- var aboveTheFoldHeight = screenHeight - overviewHeight;
- element.css('margin-top', aboveTheFoldHeight + 'px');
+ var init = function(){
+ $scope.isCookieValid();
}
- return {
- link: link
- };
- }
+ // Helpers to debug
+ // You can also use `document.cookie` in the browser dev console.
+ //console.log($cookies.getAll());
- /**
- * Fixes autofocus for inputs that are inside modals
- *
- */
- focusInput.$inject = ['$timeout'];
- function focusInput($timeout) {
- function link(scope, elem) {
- $timeout(function() {
- elem.focus();
- });
+ $scope.isCookieValid = function() {
+ // Use a boolean for the ng-hide, because using a function with ng-hide
+ // is considered bad practice. The digest cycle will call it multiple
+ // times, in our case around 240 times.
+ $scope.isCookieValidBool = ($cookies.get('consent') === 'true')
}
- return {
- link: link
+
+ $scope.acceptCookie = function() {
+ //console.log('Accepting cookie...');
+ var today = new Date();
+ var expireDate = new Date(today);
+ expireDate.setMonth(today.getMonth() + 6);
+
+ $cookies.put('consent', true, {'expires' : expireDate.toUTCString()} );
+
+ // Trigger the check again, after we click
+ $scope.isCookieValid();
};
+
+ init();
+
}
+ };
+}
+
+
})();
(function() {
'use strict';
angular.module('app.components')
- .directive('activeButton', activeButton);
+ .directive('chart', chart);
- /**
- * Used to highlight and unhighlight buttons on kit menu
- *
- * It attaches click handlers dynamically
- */
+ chart.$inject = ['sensor', 'animation', '$timeout', '$window'];
+ function chart(sensor, animation, $timeout, $window) {
+ var margin, width, height, svg, xScale, yScale0, yScale1, xAxis, yAxisLeft, yAxisRight, dateFormat, areaMain, valueLineMain, areaCompare, valueLineCompare, focusCompare, focusMain, popup, dataMain, colorMain, yAxisScale, unitMain, popupContainer;
- activeButton.$inject = ['$timeout', '$window'];
- function activeButton($timeout, $window) {
return {
link: link,
- restrict: 'A'
-
+ restrict: 'A',
+ scope: {
+ chartData: '='
+ }
};
- ////////////////////////////
-
- function link(scope, element) {
- var childrens = element.children();
- var container;
+ function link(scope, elem) {
$timeout(function() {
- var navbar = angular.element('.stickNav');
- var kitMenu = angular.element('.kit_menu');
- var kitOverview = angular.element('.kit_overview');
- var kitDashboard = angular.element('.kit_chart');
- var kitDetails = angular.element('.kit_details');
- var kitOwner = angular.element('.kit_owner');
- var kitComments = angular.element('.kit_comments');
-
- container = {
- navbar: {
- height: navbar.height()
- },
- kitMenu: {
- height: kitMenu.height()
- },
- kitOverview: {
- height: kitOverview.height(),
- offset: kitOverview.offset().top,
- buttonOrder: 0
- },
- kitDashboard: {
- height: kitDashboard.height(),
- offset: kitDashboard.offset().top,
- buttonOrder: 40
- },
- kitDetails: {
- height: kitDetails.height(),
- offset: kitDetails.offset() ? kitDetails.offset().top : 0,
- buttonOrder: 1
- },
- kitOwner: {
- height: kitOwner.height(),
- offset: kitOwner.offset() ? kitOwner.offset().top : 0,
- buttonOrder: 2
- },
- kitComments: {
- height: kitComments.height(),
- offset: kitComments.offset() ? kitComments.offset().top : 0,
- buttonOrder: 3
- }
- };
- }, 1000);
-
- function scrollTo(offset) {
- if(!container) {
- return;
- }
- angular.element($window).scrollTop(offset - container.navbar.height - container.kitMenu.height);
- }
-
- function getButton(buttonOrder) {
- return childrens[buttonOrder];
- }
+ createChart(elem[0]);
+ }, 0);
- function unHighlightButtons() {
- //remove border, fill and stroke of every icon
- var activeButton = angular.element('.md-button.button_active');
- if(activeButton.length) {
- activeButton.removeClass('button_active');
+ var lastData = {};
- var strokeContainer = activeButton.find('.stroke_container');
- strokeContainer.css('stroke', 'none');
- strokeContainer.css('stroke-width', '1');
+ // on window resize, it re-renders the chart to fit into the new window size
+ angular.element($window).on('resize', function() {
+ createChart(elem[0]);
+ updateChartData(lastData.data, {type: lastData.type, container: elem[0], color: lastData.color, unit: lastData.unit});
+ });
- var fillContainer = strokeContainer.find('.fill_container');
- fillContainer.css('fill', '#FF8600');
+ scope.$watch('chartData', function(newData) {
+ if(!newData) {
+ return;
}
- }
- function highlightButton(button) {
- var clickedButton = angular.element(button);
- //add border, fill and stroke to every icon
- clickedButton.addClass('button_active');
+ if(newData !== undefined) {
+ // if there's data for 2 sensors
+ if(newData[0] && newData[1]) {
+ var sensorDataMain = newData[0].data;
+ // we could get some performance from saving the map in the showKit controller on line 218 and putting that logic in here
+ var dataMain = sensorDataMain.map(function(dataPoint) {
+ return {
+ date: dateFormat(dataPoint.time),
+ count: dataPoint && dataPoint.count,
+ value: dataPoint && dataPoint.value
+ };
+ });
+ // sort data points by date
+ dataMain.sort(function(a, b) {
+ return a.date - b.date;
+ });
- var strokeContainer = clickedButton.find('.stroke_container');
- strokeContainer.css('stroke', 'white');
- strokeContainer.css('stroke-width', '0.01px');
+ var sensorDataCompare = newData[1].data;
+ var dataCompare = sensorDataCompare.map(function(dataPoint) {
+ return {
+ date: dateFormat(dataPoint.time),
+ count: dataPoint && dataPoint.count,
+ value: dataPoint && dataPoint.value
+ };
+ });
- var fillContainer = strokeContainer.find('.fill_container');
- fillContainer.css('fill', 'white');
- }
+ dataCompare.sort(function(a, b) {
+ return a.date - b.date;
+ });
- //attach event handlers for clicks for every button and scroll to a section when clicked
- _.each(childrens, function(button) {
- angular.element(button).on('click', function() {
- var buttonOrder = angular.element(this).index();
- for(var elem in container) {
- if(container[elem].buttonOrder === buttonOrder) {
- var offset = container[elem].offset;
- scrollTo(offset);
- angular.element($window).trigger('scroll');
- }
- }
- });
- });
+ var data = [dataMain, dataCompare];
+ var colors = [newData[0].color, newData[1].color];
+ var units = [newData[0].unit, newData[1].unit];
+ // saves everything in case we need to re-render
+ lastData = {
+ data: data,
+ type: 'both',
+ color: colors,
+ unit: units
+ };
+ // call function to update the chart with the new data
+ updateChartData(data, {type: 'both', container: elem[0], color: colors, unit: units });
+ // if only data for the main sensor
+ } else if(newData[0]) {
- var currentSection;
+ var sensorData = newData[0].data;
+ /*jshint -W004 */
+ var data = sensorData.map(function(dataPoint) {
+ return {
+ date: dateFormat(dataPoint.time),
+ count: dataPoint && dataPoint.count,
+ value: dataPoint && dataPoint.value
+ };
+ });
- //on scroll, check if window is on a section
- angular.element($window).on('scroll', function() {
- if(!container){ return; }
+ data.sort(function(a, b) {
+ return a.date - b.date;
+ });
- var windowPosition = document.body.scrollTop;
- var appPosition = windowPosition + container.navbar.height + container.kitMenu.height;
- var button;
- if(currentSection !== 'none' && appPosition <= container.kitOverview.offset) {
- button = getButton(container.kitOverview.buttonOrder);
- unHighlightButtons();
- currentSection = 'none';
- } else if(currentSection !== 'overview' && appPosition >= container.kitOverview.offset && appPosition <= container.kitOverview.offset + container.kitOverview.height) {
- button = getButton(container.kitOverview.buttonOrder);
- unHighlightButtons();
- highlightButton(button);
- currentSection = 'overview';
- } else if(currentSection !== 'details' && appPosition >= container.kitDetails.offset && appPosition <= container.kitDetails.offset + container.kitDetails.height) {
- button = getButton(container.kitDetails.buttonOrder);
- unHighlightButtons();
- highlightButton(button);
- currentSection = 'details';
- } else if(currentSection !== 'owner' && appPosition >= container.kitOwner.offset && appPosition <= container.kitOwner.offset + container.kitOwner.height) {
- button = getButton(container.kitOwner.buttonOrder);
- unHighlightButtons();
- highlightButton(button);
- currentSection = 'owner';
- } else if(currentSection !== 'comments' && appPosition >= container.kitComments.offset && appPosition <= container.kitComments.offset + container.kitOwner.height) {
- button = getButton(container.kitComments.buttonOrder);
- unHighlightButtons();
- highlightButton(button);
- currentSection = 'comments';
+ var color = newData[0].color;
+ var unit = newData[0].unit;
+
+ lastData = {
+ data: data,
+ type: 'main',
+ color: color,
+ unit: unit
+ };
+
+ updateChartData(data, {type: 'main', container: elem[0], color: color, unit: unit });
+ }
+ animation.hideChartSpinner();
}
});
}
- }
-})();
-(function() {
- 'use strict';
+ // creates the container that is re-used across different sensor charts
+ function createChart(elem) {
+ d3.select(elem).selectAll('*').remove();
- angular.module('app.components')
- .factory('user', user);
+ margin = {top: 20, right: 12, bottom: 20, left: 42};
+ width = elem.clientWidth - margin.left - margin.right;
+ height = elem.clientHeight - margin.top - margin.bottom;
- user.$inject = ['Restangular'];
- function user(Restangular) {
- var service = {
- createUser: createUser,
- getUser: getUser,
- updateUser: updateUser
- };
- return service;
+ xScale = d3.time.scale().range([0, width]);
+ xScale.tickFormat("%Y-%m-%d %I:%M:%S");
+ yScale0 = d3.scale.linear().range([height, 0]);
+ yScale1 = d3.scale.linear().range([height, 0]);
+ yAxisScale = d3.scale.linear().range([height, 0]);
- ////////////////////
+ dateFormat = d3.time.format('%Y-%m-%dT%H:%M:%S').parse;//d3.time.format('%Y-%m-%dT%X.%LZ').parse; //'YYYY-MM-DDTHH:mm:ssZ'
- function createUser(signupData) {
- return Restangular.all('users').post(signupData);
- }
+ xAxis = d3.svg.axis()
+ .scale(xScale)
+ .orient('bottom')
+ .ticks(5);
- function getUser(id) {
- return Restangular.one('users', id).get();
- }
+ yAxisLeft = d3.svg.axis()
+ .scale(yScale0)
+ .orient('left')
+ .ticks(5);
- function updateUser(updateData) {
- return Restangular.all('me').customPUT(updateData);
+ yAxisRight = d3.svg.axis()
+ .scale(yScale1)
+ .orient('right')
+ .ticks(5);
+
+ areaMain = d3.svg.area()
+ .defined(function(d) {return d.value != null })
+ .interpolate('linear')
+ .x(function(d) { return xScale(d.date); })
+ .y0(height)
+ .y1(function(d) { return yScale0(d.count); });
+
+ valueLineMain = d3.svg.line()
+ .defined(function(d) {return d.value != null })
+ .interpolate('linear')
+ .x(function(d) { return xScale(d.date); })
+ .y(function(d) { return yScale0(d.count); });
+
+ areaCompare = d3.svg.area()
+ .defined(function(d) {return d.value != null })
+ .interpolate('linear')
+ .x(function(d) { return xScale(d.date); })
+ .y0(height)
+ .y1(function(d) { return yScale1(d.count); });
+
+ valueLineCompare = d3.svg.line()
+ .defined(function(d) {return d.value != null })
+ .interpolate('linear')
+ .x(function(d) { return xScale(d.date); })
+ .y(function(d) { return yScale1(d.count); });
+
+ svg = d3
+ .select(elem)
+ .append('svg')
+ .attr('width', width + margin.left + margin.right)
+ .attr('height', height + margin.top + margin.bottom)
+ .append('g')
+ .attr('transform', 'translate(' + (margin.left - margin.right) + ',' + margin.top + ')');
}
- }
-})();
+ // calls functions depending on type of chart
+ function updateChartData(newData, options) {
+ if(options.type === 'main') {
+ updateChartMain(newData, options);
+ } else if(options.type === 'both') {
+ updateChartCompare(newData, options);
+ }
+ }
+ // function in charge of rendering when there's data for 1 sensor
+ function updateChartMain(data, options) {
+ xScale.domain(d3.extent(data, function(d) { return d.date; }));
+ yScale0.domain([(d3.min(data, function(d) { return d.count; })) * 0.8, (d3.max(data, function(d) { return d.count; })) * 1.2]);
-(function() {
- 'use strict';
+ svg.selectAll('*').remove();
- angular.module('app.components')
- .factory('tag', tag);
+ //Add the area path
+ svg.append('path')
+ .datum(data)
+ .attr('class', 'chart_area')
+ .attr('fill', options.color)
+ .attr('d', areaMain);
- tag.$inject = ['Restangular'];
- function tag(Restangular) {
- var tags = [];
- var selectedTags = [];
+ // Add the valueline path.
+ svg.append('path')
+ .attr('class', 'chart_line')
+ .attr('stroke', options.color)
+ .attr('d', valueLineMain(data));
- var service = {
- getTags: getTags,
- getSelectedTags: getSelectedTags,
- setSelectedTags: setSelectedTags,
- tagWithName: tagWithName,
- filterMarkersByTag: filterMarkersByTag
- };
+ // Add the X Axis
+ svg.append('g')
+ .attr('class', 'axis x')
+ .attr('transform', 'translate(0,' + height + ')')
+ .call(xAxis);
- return service;
+ // Add the Y Axis
+ svg.append('g')
+ .attr('class', 'axis y_left')
+ .call(yAxisLeft);
- /////////////////
+ // Draw the x Grid lines
+ svg.append('g')
+ .attr('class', 'grid')
+ .attr('transform', 'translate(0,' + height + ')')
+ .call(xGrid()
+ .tickSize(-height, 0, 0)
+ .tickFormat('')
+ );
- function getTags() {
- return Restangular.all('tags')
- .getList({'per_page': 200})
- .then(function(fetchedTags){
- tags = fetchedTags.plain();
- return tags;
- });
- }
+ // Draw the y Grid lines
+ svg.append('g')
+ .attr('class', 'grid')
+ .call(yGrid()
+ .tickSize(-width, 0, 0)
+ .tickFormat('')
+ );
- function getSelectedTags(){
- return selectedTags;
- }
-
- function setSelectedTags(tags){
- selectedTags = tags;
- }
+ focusMain = svg.append('g')
+ .attr('class', 'focus')
+ .style('display', 'none');
- function tagWithName(name){
- var result = _.where(tags, {name: name});
- if (result && result.length > 0){
- return result[0];
- }else{
- return;
- }
- }
+ focusMain.append('circle')
+ .style('stroke', options.color)
+ .attr('r', 4.5);
- function filterMarkersByTag(tmpMarkers) {
- var markers = filterMarkers(tmpMarkers);
- return markers;
- }
+ var popupWidth = 84;
+ var popupHeight = 46;
- function filterMarkers(tmpMarkers) {
- if (service.getSelectedTags().length === 0){
- return tmpMarkers;
- }
- return tmpMarkers.filter(function(marker) {
- var tags = marker.myData.tags;
- if (tags.length === 0){
- return false;
- }
- return _.some(tags, function(tag) {
- return _.includes(service.getSelectedTags(), tag);
- });
- });
- }
- }
-})();
+ popup = svg.append('g')
+ .attr('class', 'focus')
+ .style('display', 'none');
-(function() {
- 'use strict';
+ popupContainer = popup.append('rect')
+ .attr('width', popupWidth)
+ .attr('height', popupHeight)
+ .attr('transform', function() {
+ var result = 'translate(-42, 5)';
- angular.module('app.components')
- .factory('sensor', sensor);
+ return result;
+ })
+ .style('stroke', 'grey')
+ .style('stroke-width', '0.5')
+ .style('fill', 'white');
- sensor.$inject = ['Restangular', 'timeUtils', 'sensorUtils'];
- function sensor(Restangular, timeUtils, sensorUtils) {
- var sensorTypes;
- callAPI().then(function(data) {
- setTypes(data);
- });
+ var text = popup.append('text')
+ .attr('class', '');
+
+ var textMain = text.append('tspan')
+ .attr('class', 'popup_main')
+ .attr('text-anchor', 'start')
+ .attr('x', -popupWidth / 2)
+ .attr('dx', 8)
+ .attr('y', popupHeight / 2)
+ .attr('dy', 3);
+
+ textMain.append('tspan')
+ .attr('class', 'popup_value');
+
+ textMain.append('tspan')
+ .attr('class', 'popup_unit')
+ .attr('dx', 5);
+
+ text.append('tspan')
+ .attr('class', 'popup_date')
+ .attr('x', -popupWidth / 2)
+ .attr('dx', 8)
+ .attr('y', popupHeight - 2)
+ .attr('dy', 0)
+ .attr( 'text-anchor', 'start' );
- var service = {
- callAPI: callAPI,
- setTypes: setTypes,
- getTypes: getTypes,
- getSensorsData: getSensorsData
- };
- return service;
+ svg.append('rect')
+ .attr('class', 'overlay')
+ .attr('width', width)
+ .attr('height', height)
+ .on('mouseover', function() {
+ popup.style('display', null);
+ focusMain.style('display', null);
+ })
+ .on('mouseout', function() {
+ popup.style('display', 'none');
+ focusMain.style('display', 'none');
+ })
+ .on('mousemove', mousemove);
- ////////////////
- function callAPI() {
- return Restangular.all('sensors').getList({'per_page': 1000});
- }
- function setTypes(sensorTypes) {
- sensorTypes = sensorTypes;
- }
+ function mousemove() {
+ var bisectDate = d3.bisector(function(d) { return d.date; }).left;
- function getTypes() {
- return sensorTypes;
- }
+ var x0 = xScale.invert(d3.mouse(this)[0]);
+ var i = bisectDate(data, x0, 1);
+ var d0 = data[i - 1];
+ var d1 = data[i];
+ var d = d1 && (x0 - d0.date > d1.date - x0) ? d1 : d0;
- function getSensorsData(deviceID, sensorID, dateFrom, dateTo) {
- var rollup = sensorUtils.getRollup(dateFrom, dateTo);
- dateFrom = timeUtils.convertTime(dateFrom);
- dateTo = timeUtils.convertTime(dateTo);
+ focusMain.attr('transform', 'translate(' + xScale(d.date) + ', ' + yScale0(d.count) + ')');
+ var popupText = popup.select('text');
+ var textMain = popupText.select('.popup_main');
+ var valueMain = textMain.select('.popup_value').text(parseValue(d.value));
+ var unitMain = textMain.select('.popup_unit').text(options.unit);
+ var date = popupText.select('.popup_date').text(parseTime(d.date));
- return Restangular.one('devices', deviceID).customGET('readings', {'from': dateFrom, 'to': dateTo, 'rollup': rollup, 'sensor_id': sensorID, 'all_intervals': true});
- }
- }
-})();
+ var textContainers = [
+ textMain,
+ date
+ ];
-(function() {
- 'use strict';
+ var popupWidth = resizePopup(popupContainer, textContainers);
- angular.module('app.components')
- .factory('search', search);
-
- search.$inject = ['$http', 'Restangular'];
- function search($http, Restangular) {
- var service = {
- globalSearch: globalSearch
- };
+ if(xScale(d.date) + 80 + popupWidth > options.container.clientWidth) {
+ popup.attr('transform', 'translate(' + (xScale(d.date) - 120) + ', ' + (d3.mouse(this)[1] - 20) + ')');
+ } else {
+ popup.attr('transform', 'translate(' + (xScale(d.date) + 80) + ', ' + (d3.mouse(this)[1] - 20) + ')');
+ }
+ }
+ }
- return service;
+ // function in charge of rendering when there's data for 2 sensors
+ function updateChartCompare(data, options) {
+ xScale.domain(d3.extent(data[0], function(d) { return d.date; }));
+ yScale0.domain([(d3.min(data[0], function(d) { return d.count; })) * 0.8, (d3.max(data[0], function(d) { return d.count; })) * 1.2]);
+ yScale1.domain([(d3.min(data[1], function(d) { return d.count; })) * 0.8, (d3.max(data[1], function(d) { return d.count; })) * 1.2]);
- /////////////////////////
+ svg.selectAll('*').remove();
- function globalSearch(query) {
- return Restangular.all('search').getList({q: query});
- }
- }
-})();
+ //Add both area paths
+ svg.append('path')
+ .datum(data[0])
+ .attr('class', 'chart_area')
+ .attr('fill', options.color[0])
+ .attr('d', areaMain);
-(function() {
- 'use strict';
+ svg.append('path')
+ .datum(data[1])
+ .attr('class', 'chart_area')
+ .attr('fill', options.color[1])
+ .attr('d', areaCompare);
- angular.module('app.components')
- .factory('measurement', measurement);
+ // Add both valueline paths.
+ svg.append('path')
+ .attr('class', 'chart_line')
+ .attr('stroke', options.color[0])
+ .attr('d', valueLineMain(data[0]));
- measurement.$inject = ['Restangular'];
+ svg.append('path')
+ .attr('class', 'chart_line')
+ .attr('stroke', options.color[1])
+ .attr('d', valueLineCompare(data[1]));
- function measurement(Restangular) {
+ // Add the X Axis
+ svg.append('g')
+ .attr('class', 'axis x')
+ .attr('transform', 'translate(0,' + height + ')')
+ .call(xAxis);
- var service = {
- getTypes: getTypes,
- getMeasurement: getMeasurement
+ // Add both Y Axis
+ svg.append('g')
+ .attr('class', 'axis y_left')
+ .call(yAxisLeft);
- };
- return service;
+ svg.append('g')
+ .attr('class', 'axis y_right')
+ .attr('transform', 'translate(' + width + ' ,0)')
+ .call(yAxisRight);
- ////////////////
+ // Draw the x Grid lines
+ svg.append('g')
+ .attr('class', 'grid')
+ .attr('transform', 'translate(0,' + height + ')')
+ .call(xGrid()
+ .tickSize(-height, 0, 0)
+ .tickFormat('')
+ );
+ // Draw the y Grid lines
+ svg.append('g')
+ .attr('class', 'grid')
+ .call(yGrid()
+ .tickSize(-width, 0, 0)
+ .tickFormat('')
+ );
- function getTypes() {
- return Restangular.all('measurements').getList({'per_page': 1000});
- }
+ focusCompare = svg.append('g')
+ .attr('class', 'focus')
+ .style('display', 'none');
- function getMeasurement(mesID) {
+ focusMain = svg.append('g')
+ .attr('class', 'focus')
+ .style('display', 'none');
- return Restangular.one('measurements', mesID).get();
- }
- }
-})();
-(function() {
- 'use strict';
+ focusCompare.append('circle')
+ .style('stroke', options.color[1])
+ .attr('r', 4.5);
- angular.module('app.components')
- .factory('geolocation', geolocation);
+ focusMain.append('circle')
+ .style('stroke', options.color[0])
+ .attr('r', 4.5);
- geolocation.$inject = ['$http', '$window'];
- function geolocation($http, $window) {
+ var popupWidth = 84;
+ var popupHeight = 75;
- var service = {
- grantHTML5Geolocation: grantHTML5Geolocation,
- isHTML5GeolocationGranted: isHTML5GeolocationGranted
- };
- return service;
+ popup = svg.append('g')
+ .attr('class', 'focus')
+ .style('display', 'none');
- ///////////////////////////
+ popupContainer = popup.append('rect')
+ .attr('width', popupWidth)
+ .attr('height', popupHeight)
+ .style('min-width', '40px')
+ .attr('transform', function() {
+ var result = 'translate(-42, 5)';
+ return result;
+ })
+ .style('stroke', 'grey')
+ .style('stroke-width', '0.5')
+ .style('fill', 'white');
- function grantHTML5Geolocation(){
- $window.localStorage.setItem('smartcitizen.geolocation_granted', true);
- }
+ popup.append('rect')
+ .attr('width', 8)
+ .attr('height', 2)
+ .attr('transform', function() {
+ return 'translate(' + (-popupWidth / 2 + 4).toString() + ', 20)';
+ })
+ .style('fill', options.color[0]);
- function isHTML5GeolocationGranted(){
- return $window.localStorage
- .getItem('smartcitizen.geolocation_granted');
- }
- }
-})();
+ popup.append('rect')
+ .attr('width', 8)
+ .attr('height', 2)
+ .attr('transform', function() {
+ return 'translate(' + (-popupWidth / 2 + 4).toString() + ', 45)';
+ })
+ .style('fill', options.color[1]);
-(function() {
- 'use strict';
+ var text = popup.append('text')
+ .attr('class', '');
- angular.module('app.components')
- .factory('file', file);
+ var textMain = text.append('tspan')
+ .attr('class', 'popup_main')
+ .attr('x', -popupHeight / 2 + 7) //position of text
+ .attr('dx', 8) //margin given to the element, will be applied to both sides thanks to resizePopup function
+ .attr('y', popupHeight / 3)
+ .attr('dy', 3);
- file.$inject = ['Restangular', 'Upload'];
- function file(Restangular, Upload) {
- var service = {
- getCredentials: getCredentials,
- uploadFile: uploadFile,
- getImageURL: getImageURL
- };
- return service;
+ textMain.append('tspan')
+ .attr('class', 'popup_value')
+ .attr( 'text-anchor', 'start' );
- ///////////////
+ textMain.append('tspan')
+ .attr('class', 'popup_unit')
+ .attr('dx', 5);
- function getCredentials(filename) {
- var data = {
- filename: filename
- };
- return Restangular.all('me/avatar').post(data);
- }
+ var textCompare = text.append('tspan')
+ .attr('class', 'popup_compare')
+ .attr('x', -popupHeight / 2 + 7) //position of text
+ .attr('dx', 8) //margin given to the element, will be applied to both sides thanks to resizePopup function
+ .attr('y', popupHeight / 1.5)
+ .attr('dy', 3);
- function uploadFile(fileData, key, policy, signature) {
- return Upload.upload({
- url: 'https://smartcitizen.s3-eu-west-1.amazonaws.com',
- method: 'POST',
- data: {
- key: key,
- policy: policy,
- signature: signature,
- AWSAccessKeyId: 'AKIAJ753OQI6JPSDCPHA',
- acl: 'public-read',
- "Content-Type": fileData.type || 'application/octet-stream',
- /*jshint camelcase: false */
- success_action_status: 200,
- file: fileData
- }
- });
- }
+ textCompare.append('tspan')
+ .attr('class', 'popup_value')
+ .attr( 'text-anchor', 'start' );
- function getImageURL(filename, size) {
- size = size === undefined ? 's101' : size;
+ textCompare.append('tspan')
+ .attr('class', 'popup_unit')
+ .attr('dx', 5);
- return 'https://images.smartcitizen.me/' + size + '/' + filename;
- }
- }
-})();
+ text.append('tspan')
+ .attr('class', 'popup_date')
+ .attr('x', (- popupWidth / 2))
+ .attr('dx', 8)
+ .attr('y', popupHeight - 2)
+ .attr('dy', 0)
+ .attr( 'text-anchor', 'start' );
-(function() {
- 'use strict';
+ svg.append('rect')
+ .attr('class', 'overlay')
+ .attr('width', width)
+ .attr('height', height)
+ .on('mouseover', function() {
+ focusCompare.style('display', null);
+ focusMain.style('display', null);
+ popup.style('display', null);
+ })
+ .on('mouseout', function() {
+ focusCompare.style('display', 'none');
+ focusMain.style('display', 'none');
+ popup.style('display', 'none');
+ })
+ .on('mousemove', mousemove);
- angular.module('app.components')
- .factory('device', device);
+ function mousemove() {
+ var bisectDate = d3.bisector(function(d) { return d.date; }).left;
- device.$inject = ['Restangular', '$window', 'timeUtils','$http', 'auth', '$rootScope'];
- function device(Restangular, $window, timeUtils, $http, auth, $rootScope) {
- var worldMarkers;
+ var x0 = xScale.invert(d3.mouse(this)[0]);
+ var i = bisectDate(data[1], x0, 1);
+ var d0 = data[1][i - 1];
+ var d1 = data[1][i];
+ var d = x0 - d0.date > d1.date - x0 ? d1 : d0;
+ focusCompare.attr('transform', 'translate(' + xScale(d.date) + ', ' + yScale1(d.count) + ')');
- initialize();
- var service = {
- getDevices: getDevices,
- getAllDevices: getAllDevices,
- getDevice: getDevice,
- createDevice: createDevice,
- updateDevice: updateDevice,
- getWorldMarkers: getWorldMarkers,
- setWorldMarkers: setWorldMarkers,
- mailReadings: mailReadings,
- postReadings: postReadings,
- removeDevice: removeDevice,
- updateContext: updateContext
- };
+ var dMain0 = data[0][i - 1];
+ var dMain1 = data[0][i];
+ var dMain = x0 - dMain0.date > dMain1.date - x0 ? dMain1 : dMain0;
+ focusMain.attr('transform', 'translate(' + xScale(dMain.date) + ', ' + yScale0(dMain.count) + ')');
- return service;
+ var popupText = popup.select('text');
+ var textMain = popupText.select('.popup_main');
+ textMain.select('.popup_value').text(parseValue(dMain.value));
+ textMain.select('.popup_unit').text(options.unit[0]);
+ var textCompare = popupText.select('.popup_compare');
+ textCompare.select('.popup_value').text(parseValue(d.value));
+ textCompare.select('.popup_unit').text(options.unit[1]);
+ var date = popupText.select('.popup_date').text(parseTime(d.date));
- //////////////////////////
+ var textContainers = [
+ textMain,
+ textCompare,
+ date
+ ];
- function initialize() {
- if(areMarkersOld()) {
- removeMarkers();
+ var popupWidth = resizePopup(popupContainer, textContainers);
+
+ if(xScale(d.date) + 80 + popupWidth > options.container.clientWidth) {
+ popup.attr('transform', 'translate(' + (xScale(d.date) - 120) + ', ' + (d3.mouse(this)[1] - 20) + ')');
+ } else {
+ popup.attr('transform', 'translate(' + (xScale(d.date) + 80) + ', ' + (d3.mouse(this)[1] - 20) + ')');
+ }
}
}
- function getDevices(location) {
- var parameter = '';
- parameter += location.lat + ',' + location.lng;
- return Restangular.all('devices').getList({near: parameter, 'per_page': '100'});
+ function xGrid() {
+ return d3.svg.axis()
+ .scale(xScale)
+ .orient('bottom')
+ .ticks(5);
}
- function getAllDevices(forceReload) {
- if (forceReload || auth.isAuth()) {
- return getAllDevicesNoCached();
- } else {
- return getAllDevicesCached();
- }
+ function yGrid() {
+ return d3.svg.axis()
+ .scale(yScale0)
+ .orient('left')
+ .ticks(5);
}
- function getAllDevicesCached() {
- return Restangular.all('devices/world_map')
- .getList()
- .then(function(fetchedDevices){
- return fetchedDevices.plain();
- });
+ function parseValue(value) {
+ if(value === null) {
+ return 'No data on the current timespan';
+ } else if(value.toString().indexOf('.') !== -1) {
+ var result = value.toString().split('.');
+ return result[0] + '.' + result[1].slice(0, 2);
+ } else if(value > 99.99) {
+ return value.toString();
+ } else {
+ return value.toString().slice(0, 2);
+ }
}
- function getAllDevicesNoCached() {
- return Restangular.all('devices/fresh_world_map')
- .getList()
- .then(function(fetchedDevices){
- return fetchedDevices.plain();
- });
+ function parseTime(time) {
+ return moment(time).format('h:mm a ddd Do MMM YYYY');
}
- function getDevice(id) {
- return Restangular.one('devices', id).get();
- }
+ function resizePopup(popupContainer, textContainers) {
+ if(!textContainers.length) {
+ return;
+ }
- function createDevice(data) {
- return Restangular.all('devices').post(data);
- }
+ var widestElem = textContainers.reduce(function(widestElemSoFar, textContainer) {
+ var currentTextContainerWidth = getContainerWidth(textContainer);
+ var prevTextContainerWidth = getContainerWidth(widestElemSoFar);
+ return prevTextContainerWidth >= currentTextContainerWidth ? widestElemSoFar : textContainer;
+ }, textContainers[0]);
- function updateDevice(id, data) {
- return Restangular.one('devices', id).patch(data);
- }
+ var margins = widestElem.attr('dx') * 2;
- function getWorldMarkers() {
- return worldMarkers || ($window.localStorage.getItem('smartcitizen.markers') && JSON.parse($window.localStorage.getItem('smartcitizen.markers') ).data);
- }
+ popupContainer
+ .attr('width', getContainerWidth(widestElem) + margins);
- function setWorldMarkers(data) {
- var obj = {
- timestamp: new Date(),
- data: data
- };
- try {
- $window.localStorage.setItem('smartcitizen.markers', JSON.stringify(obj) );
- } catch (e) {
- console.log("Could not store markers in localstorage. skipping...");
+ function getContainerWidth(container) {
+ var node = container.node();
+ var width;
+ if(node.getComputedTextLength) {
+ width = node.getComputedTextLength();
+ } else if(node.getBoundingClientRect) {
+ width = node.getBoundingClientRect().width;
+ } else {
+ width = node.getBBox().width;
+ }
+ return width;
}
- worldMarkers = obj.data;
+ return getContainerWidth(widestElem) + margins;
}
+ }
- function getTimeStamp() {
- return ($window.localStorage.getItem('smartcitizen.markers') &&
- JSON.parse($window.localStorage
- .getItem('smartcitizen.markers') ).timestamp);
- }
+})();
- function areMarkersOld() {
- var markersDate = getTimeStamp();
- return !timeUtils.isWithin(1, 'minutes', markersDate);
- }
+(function(){
+ 'use strict';
- function removeMarkers() {
- worldMarkers = null;
- $window.localStorage.removeItem('smartcitizen.markers');
- }
+ angular.module('app.components')
+ .directive('apiKey', apiKey);
- function mailReadings(kit) {
- return Restangular
- .one('devices', kit.id)
- .customGET('readings/csv_archive');
- }
+ function apiKey(){
+ return {
+ scope: {
+ apiKey: '=apiKey'
+ },
+ restrict: 'A',
+ controller: 'ApiKeyController',
+ controllerAs: 'vm',
+ templateUrl: 'app/components/apiKey/apiKey.html'
+ };
+ }
+})();
- function postReadings(kit, readings) {
- return Restangular
- .one('devices', kit.id)
- .post('readings', readings);
- }
+(function(){
+ 'use strict';
+
+ angular.module('app.components')
+ .controller('ApiKeyController', ApiKeyController);
+
+ ApiKeyController.$inject = ['alert'];
+ function ApiKeyController(alert){
+ var vm = this;
+
+ vm.copied = copied;
+ vm.copyFail = copyFail;
+
+ ///////////////
- function removeDevice(deviceID){
- return Restangular
- .one('devices', deviceID)
- .remove().then(function () {
- $rootScope.$broadcast('devicesContextUpdated');
- })
- ;
- }
+ function copied(){
+ alert.success('API key copied to your clipboard.');
+ }
- function updateContext (){
- return auth.updateUser().then(function(){
- removeMarkers();
- $rootScope.$broadcast('devicesContextUpdated');
- });
- }
+ function copyFail(err){
+ console.log('Copy error: ', err);
+ alert.error('Oops! An error occurred copying the api key.');
+ }
- }
+ }
})();
(function() {
'use strict';
angular.module('app.components')
- .factory('auth', auth);
-
- auth.$inject = ['$location', '$window', '$state', 'Restangular',
- '$rootScope', 'AuthUser', '$timeout', 'alert', '$cookies'];
- function auth($location, $window, $state, Restangular, $rootScope, AuthUser,
- $timeout, alert, $cookies) {
+ .factory('alert', alert);
- var user = {};
+ alert.$inject = ['$mdToast'];
+ function alert($mdToast) {
+ var service = {
+ success: success,
+ error: error,
+ info: {
+ noData: {
+ visitor: infoNoDataVisitor,
+ owner: infoNoDataOwner,
+ private: infoDataPrivate,
+ },
+ longTime: infoLongTime,
+ // TODO: Refactor, check why this was removed
+ // inValid: infoDataInvalid,
+ generic: info
+ }
+ };
- //wait until http interceptor is added to Restangular
- $timeout(function() {
- initialize();
- }, 100);
+ return service;
- var service = {
- isAuth: isAuth,
- setCurrentUser: setCurrentUser,
- getCurrentUser: getCurrentUser,
- updateUser: updateUser,
- saveToken: saveToken,
- getToken: getToken,
- login: login,
- logout: logout,
- recoverPassword: recoverPassword,
- getResetPassword: getResetPassword,
- patchResetPassword: patchResetPassword,
- isAdmin: isAdmin
- };
- return service;
+ ///////////////////
- //////////////////////////
+ function success(message) {
+ toast('success', message);
+ }
- function initialize() {
- //console.log('---- AUTH INIT -----');
- setCurrentUser('appLoad');
- }
+ function error(message) {
+ toast('error', message);
+ }
- //run on app initialization so that we can keep auth across different sessions
- // 1. Check if token in cookie exists. Return if it doesn't, user needs to login (and save a token to the cookie)
- // 2. Populate user.data with the response from the API.
- // 3. Broadcast logged in
- function setCurrentUser(time) {
- // TODO later: Should we check if token is expired here?
- if (getToken()) {
- user.token = getToken();
- }else{
- //console.log('token not found in cookie, returning');
- return;
- }
+ function infoNoDataVisitor() {
+ info('Woah! We couldn\'t locate this kit on the map because it hasn\'t published any data. Leave a ' +
+ 'comment to let its owner know.',
+ 10000,
+ {
+ button: 'Leave comment',
+ href: 'https://forum.smartcitizen.me/'
+ });
+ }
- return getCurrentUserFromAPI()
- .then(function(data) {
- // Save user.data also in localStorage. It is beeing used across the app.
- // Should it instead just be saved in the user object? Or is it OK to also have it in localStorage?
- $window.localStorage.setItem('smartcitizen.data', JSON.stringify(data.plain()) );
+ function infoNoDataOwner() {
+ info('Woah! We couldn\'t locate this kit on the map because it hasn\'t published any data.',
+ 10000);
+ }
- var newUser = new AuthUser(data);
- //check sensitive information
- if(user.data && user.data.role !== newUser.role) {
- user.data = newUser;
- $location.path('/');
- }
- user.data = newUser;
+ function infoDataPrivate() {
+ info('Device not found, or it has been set to private. Leave a ' +
+ 'comment to let its owner know you\'re interested.',
+ 10000,
+ {
+ button: 'Leave comment',
+ href: 'https://forum.smartcitizen.me/'
+ });
+ }
- //console.log('-- User populated with data: ', user)
- // Broadcast happens 2x, so the user wont think he is not logged in.
- // The 2nd broadcast waits 3sec, because f.x. on the /kits/ page, the layout has not loaded when the broadcast is sent
- $rootScope.$broadcast('loggedIn');
+ // TODO: Refactor, check why this was removed
+ // function infoDataInvalid() {
+ // info('Device not found, or it has been set to private.',
+ // 10000);
+ // }
- // used for app initialization
- if(time && time === 'appLoad') {
- //wait until navbar is loaded to emit event
- $timeout(function() {
- $rootScope.$broadcast('loggedIn', {time: 'appLoad'});
- }, 3000);
- } else {
- // used for login
- //$state.reload();
- $timeout(function() {
- alert.success('Login was successful');
- $rootScope.$broadcast('loggedIn', {});
- }, 2000);
- }
- });
- }
+ function infoLongTime() {
+ info('😅 It looks like this kit hasn\'t posted any data in a long ' +
+ 'time. Why not leave a comment to let its owner know?',
+ 10000,
+ {
+ button: 'Leave comment',
+ href: 'https://forum.smartcitizen.me/'
+ });
+ }
- // Called from device.service.js updateContext(), which is called from multiple /kit/ pages
- function updateUser() {
- return getCurrentUserFromAPI()
- .then(function(data) {
- // TODO: Should this update the token or user.data? Then it could instead call setCurrentUser?
- $window.localStorage.setItem('smartcitizen.data', JSON.stringify(data.plain()) );
- return getCurrentUser();
- });
+ function info(message, delay, options) {
+ if(options && options.button) {
+ toast('infoButton', message, options, undefined, delay);
+ } else {
+ toast('info', message, options, undefined, delay);
}
+ }
- function getCurrentUser() {
- user.token = getToken();
- user.data = $window.localStorage.getItem('smartcitizen.data') && new AuthUser(JSON.parse( $window.localStorage.getItem('smartcitizen.data') ));
- return user;
- }
+ function toast(type, message, options, position, delay) {
+ position = position === undefined ? 'top': position;
+ delay = delay === undefined ? 5000 : delay;
- // Should check if user.token exists - but now checks if the cookies.token exists.
- function isAuth() {
- // TODO: isAuth() is called from many different services BEFORE auth.init has run.
- // That means that the user.token is EMPTY, meaning isAuth will be false
- // We can cheat and just check the cookie, but we should NOT. Because auth.init should also check if the cookie is valid / expired
- // Ideally it should return !!user.token
- //return !!user.token;
- return !!getToken();
- }
+ $mdToast.show({
+ controller: 'AlertController',
+ controllerAs: 'vm',
+ templateUrl: 'app/components/alert/alert' + type + '.html',
+ hideDelay: delay,
+ position: position,
+ locals: {
+ message: message,
+ button: options && options.button,
+ href: options && options.href
+ }
+ });
+ }
+ }
+})();
- // LoginModal calls this after it receives the token from the API, and wants to save it in a cookie.
- function saveToken(token) {
- //console.log('saving Token to cookie:', token);
- $cookies.put('smartcitizen.token', token);
- setCurrentUser();
- }
+(function() {
+ 'use strict';
- function getToken(){
- return $cookies.get('smartcitizen.token');
- }
+ angular.module('app.components')
+ .controller('AlertController', AlertController);
- function login(loginData) {
- return Restangular.all('sessions').post(loginData);
- }
+ AlertController.$inject = ['$scope', '$mdToast', 'message', 'button', 'href'];
+ function AlertController($scope, $mdToast, message, button, href) {
+ var vm = this;
- function logout() {
- $cookies.remove('smartcitizen.token');
- }
+ vm.close = close;
+ vm.message = message;
+ vm.button = button;
+ vm.href = href;
- function getCurrentUserFromAPI() {
- return Restangular.all('').customGET('me');
- }
+ // hideAlert will be triggered on state change
+ $scope.$on('hideAlert', function() {
+ close();
+ });
- function recoverPassword(data) {
- return Restangular.all('password_resets').post(data);
- }
+ ///////////////////
- function getResetPassword(code) {
- return Restangular.one('password_resets', code).get();
- }
- function patchResetPassword(code, data) {
- return Restangular.one('password_resets', code).patch(data);
- }
- function isAdmin(userData) {
- return userData.role === 'admin';
+ function close() {
+ $mdToast.hide();
}
}
})();
@@ -8006,7 +8006,7 @@ $templateCache.put('app/components/download/downloadModal.html','Except where otherwise noted, content on this site by Smart Citizen\xAE is licensed under a . Check the .
');
$templateCache.put('app/components/home/template.html','');
$templateCache.put('app/components/kitList/kitList.html','No kits
{{ device.name || \'No name\' }} {{ device.locationString || \'No location\' }} {{ device.hardwareName || \'Unknown Kit\'}} {{ device.state.name }}
{{ system_tag }}
View on map{{item.text}} Download CSV REMOVE
');
-$templateCache.put('app/components/landing/landing.html',' GO TO THE PLATFORM ORDER YOUR SMART CITIZEN KIT FROM SEEED STUDIO ORDER NOW WE EMPOWER COMMUNITIES TO BETTER UNDERSTAND THEIR ENVIRONMENT We\'re a community of passionate people who believe data is critical to inform political participation at all levels. We develop tools for citizen action in environmental monitoring and methodologies for community engagement and co-creation.
WATCH DOCUMENTARY
INTRODUCING A NEW AND IMPROVED KIT For the past three years, we have been working on an updated version of the Kit. The new sensors collect urban data more accurately and are easier to use. The Smart Citizen Kit 2.1 is available from Seeed studio.
ORDER NOW
TOOLS FOR EVERY COMMUNITY LOCAL COMMUNITIES Launch a crowd sensing initiative in your neighborhood. Use Smart Citizen to create local maps of noise and air quality; or use it to raise awareness and find solutions for issues that matter to your community.
RESEARCHERS Use Smart Citizen as a tool for data capture and analysis. Understand the relationship between people, environment, and technology through real-world deployments. Contribute to the project by joining the open source development community.
CITIES AND GOVERNMENTS Smart Cities should be built together with Smart Citizens. We provide the tools and knowledge to foster citizen engagement through participatory data collection, analysis and action.
CIVIC PARTICIPATION ACROSS THE GLOBE Generate real-time data and awareness about pressing environmental issues, empowering communities to seek solutions.
AN OPEN PLATFORM TO LEARN, BUILD AND CONTRIBUTE The project builds on open source technologies such as Arduino to enable citizens and communities to gather information on their environment and make it available to the public via the Smart Citizen platform.
DEVELOPER READY Use our powerful API to build amazing things using data.
USE THE API WE\u2019RE OPEN SOURCE Fork and contribute to the project in GitHub.
VISIT OUR REPOS JOIN THE FORUM A place to share ideas with the community or find support.
GET INVOLVED
');
+$templateCache.put('app/components/landing/landing.html','EMPOWERING COMMUNITIES FOR ACTION We are a collective of passionate people who believe that data is critical to inform civic participation. We develop free and open-source tools for environmental monitoring and methodologies for community-driven action.
BUILDING ON OUR HISTORY The Smart Citizen project provides tools for collecting, understanding, and sharing environmental data. Since 2012, we have developed various iterations of the Smart Citizen Kit, allowing anyone to contribute and take part in data-driven action.
GET YOUR KIT
TOOLS FOR EVERY COMMUNITY LOCAL COMMUNITIES Launch a crowd-sensing campaign in your city. Create local maps of noise and air quality. Raise awareness and find solutions for issues that matter to your community.
RESEARCH INSTITUTIONS Use Smart Citizen as a tool for environmental data collection and analysis. Trigger informed action that brings communities together, raising awareness of environmental issues.
EDUCATORS Find ways to bring Smart Citizen tools into the classroom. Explore our freely accessible tools and methodologies for educators, giving environmental awareness a hands-on toolkit.
GET YOUR KIT SMART CITIZEN KIT We\'re excited to release our newest version of the Smart Citizen Kit! Smart Citizen 2.3 is an upgrade of the kit that you already know and love, and it will be available soon through SEEED Studio. The latest version includes UV readings, an improved PM sensor and lots of new features!
Stay tuned by subscribing to our newsletter.
SUBSCRIBE
PROJECTS AND CUSTOMIZATIONS Looking for something big? Do you want to use the kit as part of a research project? While all of our hardware is open source, we work directly with research organizations and projects to make open hardware that anyone can use.
Get in touch for customized projects.
CONTACT US DEVELOPER READY Use our powerful API to build amazing things using data.
USE THE API WE\u2019RE OPEN SOURCE Fork and contribute to the project or download designs.
VISIT OUR REPOS JOIN THE FORUM A place to share ideas with the community or find support.
GET INVOLVED ');
$templateCache.put('app/components/landing/static.html','Heading 2 Heading 3 Heading 4 Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nam a porta quam. Phasellus tincidunt facilisis blandit. Aenean tempor diam quis turpis vestibulum, ac semper turpis mollis. Sed ac ultricies est. Vivamus efficitur orci efficitur turpis commodo dignissim. Aliquam sagittis risus in semper ullamcorper. Sed enim diam, tempus eget lorem sit amet, luctus porta enim. Nam aliquam mollis massa quis euismod. In commodo laoreet mattis. Nunc auctor, massa ut sollicitudin imperdiet, mauris magna tristique metus, quis lobortis ex ex id augue. In hac habitasse platea dictumst. Sed sagittis iaculis eros non sollicitudin. Sed congue, urna ut aliquet ornare, nisi tellus euismod nisi, a ullamcorper augue arcu sit amet ante. Mauris condimentum ex ante, vitae accumsan sapien vulputate in. In tempor ligula ut scelerisque feugiat. Morbi quam nisi, blandit quis malesuada sit amet, gravida ut urna.
button button Heading 2 Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nam a porta quam. Phasellus tincidunt facilisis blandit. Aenean tempor diam quis turpis vestibulum, ac semper turpis mollis. Sed ac ultricies est. Vivamus efficitur orci efficitur turpis commodo dignissim. Aliquam sagittis risus in semper ullamcorper. Sed enim diam, tempus eget lorem sit amet, luctus porta enim. Nam aliquam mollis massa quis euismod. In commodo laoreet mattis. Nunc auctor, massa ut sollicitudin imperdiet, mauris magna tristique metus, quis lobortis ex ex id augue. In hac habitasse platea dictumst. Sed sagittis iaculis eros non sollicitudin. Sed congue, urna ut aliquet ornare, nisi tellus euismod nisi, a ullamcorper augue arcu sit amet ante. Mauris condimentum ex ante, vitae accumsan sapien vulputate in. In tempor ligula ut scelerisque feugiat. Morbi quam nisi, blandit quis malesuada sit amet, gravida ut urna.
Small section Single line comment.
');
$templateCache.put('app/components/layout/layout.html','Profile Log out Map Get your kit {{item.text}}
');
$templateCache.put('app/components/login/login.html','Log In ');
@@ -8037,8 +8037,8 @@ $templateCache.put('app/components/upload/csvUpload.html','Upl
$templateCache.put('app/components/upload/errorModal.html','
Errors {{error.message}} (at row: {{error.row}}) {{csvFile.backEndErrors.statusText || csvFile.backEndErrors}} {{csvFile.backEndErrors.status}} : {{ csvFile.backEndErrors.data.message || csvFile.backEndErrors.data.errors }}
');
$templateCache.put('app/components/upload/upload.html','CSV File Upload Back to Kit
');
$templateCache.put('app/components/userProfile/userProfile.html','{{ vm.filteredDevices.length || 0 }} kits filtering by {{ vm.status.toUpperCase() || \'ALL\' }}
');
-$templateCache.put('app/core/animation/backdrop/loadingBackdrop.html',' ');
-$templateCache.put('app/core/animation/backdrop/noDataBackdrop.html','This kit hasn\u2019t still said a word \uD83D\uDC76
Your kit has still not posted any data \uD83D\uDD27\uD83D\uDD29\uD83D\uDD28 ');
$templateCache.put('app/components/kit/editKit/editKit.html','
Edit your kit
Finalise your setup
Back Save Setup your kit In order to have your kit connected to the Smart Citizen platform, we need a few step involving the connection of your kit to your computer. If this is your first time, maybe you will like to follow the Startup guide . Waiting for your kit\'s dataWe are waiting for your kit to connect on-line, this can take a few minutes Check the process on the report window and contact support@smartcitizen.me if you have any problem. Ready! Go and visit your kit on-line ');
$templateCache.put('app/components/kit/newKit/newKit.html','
Basic information Want to change your kit\'s name? Or perhaps say something nice about it in the description? Don\'t forget about the exposure!
Open data Sometimes, your devices might be collecting sensitive personal data (i.e. your exact location or by GPS using in your bike). Check the box in case you want to prevent others from accesssing your data. Manage how others can access your data:
Make this device private
Kit location Please, let us locate you, later you can adjust the location by dragging the marker on the map. Kit tags Kits can be grouped by tags. Choose from the available tags or submit a tag request on the Forum . ');
-$templateCache.put('app/components/kit/showKit/showKit.html','
No kit selected \uD83D\uDC46 Browse the map and click on any kit to see its data.
Click to see more sensors {{ sensor.value }}
{{ sensor.measurement.name }}
Click to see more sensors We can also take you to your nearest online kit by letting us know your location.
Locate me {{ vm.selectedSensorData.value }} {{ vm.selectedSensorData.unit }}
{{ sensor.measurement.name }}
This is the latest value received
{{ vm.sensorNames[vm.selectedSensor] }} {{ vm.selectedSensorData.fullDescription }}More info Compare with {{ sensor.measurement.name }}
{{ vm.device.name }} Private {{ vm.device.locationString || \'No location\' }}
{{ vm.device.hardwareName }}
{{ vm.device.state.name }}
{{ system_tag }}
Manage your kit EDIT Download CSVSD CARD UPLOAD DELETE
We empower communities to better understand their environment Smart Citizen is a project by Fab Lab Barcelona to offer an alternative to the centralised data production and management systems used by the large corporations that constitute the driving force behind the smart city concept. The project empowers citizens and communities to gather information on their environment and make it available to the public, using open source hardware and software design.
Other kits owned by {{ vm.device.owner.username }} VIEW ALL KITS BY {{ vm.device.owner.username }}
');}]);
\ No newline at end of file
+$templateCache.put('app/components/kit/showKit/showKit.html','
No kit selected \uD83D\uDC46 Browse the map and click on any kit to see its data.
Click to see more sensors {{ sensor.value }}
{{ sensor.measurement.name }}
Click to see more sensors We can also take you to your nearest online kit by letting us know your location.
Locate me {{ vm.selectedSensorData.value }} {{ vm.selectedSensorData.unit }}
{{ sensor.measurement.name }}
This is the latest value received
{{ vm.sensorNames[vm.selectedSensor] }} {{ vm.selectedSensorData.fullDescription }}More info Compare with {{ sensor.measurement.name }}
{{ vm.device.name }} Private {{ vm.device.locationString || \'No location\' }}
{{ vm.device.hardwareName }}
{{ vm.device.state.name }}
{{ system_tag }}
Manage your kit EDIT Download CSVSD CARD UPLOAD DELETE
We empower communities to better understand their environment Smart Citizen is a project by Fab Lab Barcelona to offer an alternative to the centralised data production and management systems used by the large corporations that constitute the driving force behind the smart city concept. The project empowers citizens and communities to gather information on their environment and make it available to the public, using open source hardware and software design.
Other kits owned by {{ vm.device.owner.username }} VIEW ALL KITS BY {{ vm.device.owner.username }}
');
+$templateCache.put('app/core/animation/backdrop/loadingBackdrop.html',' ');
+$templateCache.put('app/core/animation/backdrop/noDataBackdrop.html','This kit hasn\u2019t still said a word \uD83D\uDC76
Your kit has still not posted any data \uD83D\uDD27\uD83D\uDD29\uD83D\uDD28 ');}]);
\ No newline at end of file
diff --git a/styles/app-48ab85a5c6.css b/styles/app-48ab85a5c6.css
deleted file mode 100644
index 60cb6f11..00000000
--- a/styles/app-48ab85a5c6.css
+++ /dev/null
@@ -1,3009 +0,0 @@
-@import url(https://fonts.googleapis.com/css?family=Roboto+Condensed:400,300,300italic,400italic,700,700italic|Roboto:400,700,700italic,400italic);
-@import url("https://fonts.googleapis.com/css?family=Kanit:400,500,600,700,900");
-.border-black {
- border: 3px solid #1C1C1C; }
-
-.border-white {
- border: 3px solid white; }
-
-.color-white {
- color: white !important; }
-
-.color-dropdown {
- color: #7E7E7E !important; }
-
-.color-black {
- color: #1C1C1C; }
-
-.color-blue {
- color: #0019FF; }
-
-.color-cyan {
- color: #0019FF; }
-
-.color-green {
- color: #00E597; }
- .color-green svg {
- fill: #00E597; }
-
-.color-red {
- color: #FF3D4C; }
- .color-red svg {
- fill: #FF3D4C !important; }
-
-.float-left {
- float: left; }
-
-.float-right {
- float: right; }
-
-.d-flex {
- display: flex; }
-
-.full-width {
- width: 100%; }
-
-.max-width-500px {
- max-width: 500px; }
-
-.min-height-80 {
- min-height: 80%; }
-
-.font-roboto-condensed {
- font-family: 'Roboto Condensed'; }
-
-.font-kanit {
- font-family: 'Kanit'; }
-
-.text-center {
- text-align: center; }
-
-.text-left {
- text-align: left; }
-
-.bg-white {
- background-color: #fff; }
-
-.bg-black {
- background-color: black; }
-
-.bg-green {
- background-color: #00E597; }
-
-.bg-blue {
- background-color: #0019FF; }
-
-.bg-red {
- background-color: #FF3D4C !important; }
-
-.bg-red-light {
- background-color: rgba(255, 61, 76, 0.2); }
-
-.bg-yellow {
- background-color: #FFC100; }
-
-.bg-grey {
- background-color: #D2D3D5 !important; }
-
-.bg-grey-lightest {
- background-color: #F9F9FB !important; }
-
-.outline {
- outline: 1px solid red; }
-
-.btn-outline-blue {
- border: 2px solid #0019FF;
- color: #0019FF;
- padding: 10px 29px; }
- .btn-outline-blue svg {
- fill: #0019FF; }
-
-.btn-outline-blue:hover {
- background: #0019FF !important;
- color: white; }
- .btn-outline-blue:hover svg {
- fill: white; }
-
-.btn-outline-white {
- border: 2px solid white;
- color: white;
- padding: 10px 29px; }
-
-.btn-outline-white:hover {
- background: white !important;
- color: #1C1C1C; }
-
-.btn-outline-white-blue {
- border: 2px solid white;
- color: white;
- padding: 10px 29px; }
-
-.btn-outline-white-blue:hover {
- background: #0019FF !important; }
-
-.btn-outline-yellow {
- border: 2px solid #FFC100;
- color: #FFC100;
- padding: 10px 29px; }
-
-.btn-outline-yellow:hover {
- background: #FFC100 !important;
- color: #1C1C1C; }
-
-.btn-yellow {
- border: 2px solid #FFC100;
- background: #FFC100;
- color: #1C1C1C;
- padding: 12px 29px; }
-
-.btn-yellow:hover {
- background: #1C1C1C !important;
- color: #FFC100; }
-
-.btn-black-outline {
- border: 2px solid #1C1C1C;
- color: #1C1C1C;
- padding: 10px 29px; }
-
-.btn-black-outline:hover {
- background-color: #1C1C1C !important;
- color: white; }
-
-.btn-blue {
- border: 2px solid #0019FF;
- background: #0019FF;
- color: white;
- padding: 12px 29px; }
-
-.btn-blue:hover {
- color: #0019FF;
- background: #ECECEE !important; }
-
-.btn-white-blue {
- border: 2px solid white;
- background: none;
- color: white;
- padding: 12px 29px; }
-
-.btn-white-blue:hover {
- color: #0019FF;
- background: white !important;
- border: 2px solid #0019FF; }
-
-.btn-round-new {
- border-radius: 40px;
- font-family: 'Roboto Condensed';
- font-size: 16px;
- line-height: 40px;
- font-weight: 600;
- letter-spacing: 0.0325em;
- padding: 0px 35px;
- transition: box-shadow 0.4s cubic-bezier(0.25, 0.8, 0.25, 1), background-color 0.4s cubic-bezier(0.25, 0.8, 0.25, 1); }
-
-.btn-round {
- border: 0;
- border-radius: 25px;
- padding: 5px 35px;
- font-size: 18px; }
-
-.btn-cyan {
- background-color: #0019FF;
- color: white;
- border: 2px solid #0019FF; }
-
-.btn-cyan:hover {
- color: #0019FF;
- background: white !important;
- border: 2px solid #0019FF; }
-
-.btn-full {
- display: block;
- width: 100%;
- border-radius: 0px;
- margin: 0px;
- padding: 12px 29px; }
-
-.btn-small {
- min-width: unset !important;
- min-height: unset !important; }
-
-.m-0 {
- margin: 0; }
-
-.m-10 {
- margin: 10px; }
-
-.ml-0 {
- margin-left: 0px; }
-
-.ml-15 {
- margin-left: 15px; }
-
-.mt-10 {
- margin-top: 10px; }
-
-.mt-20 {
- margin-top: 20px; }
-
-.mt-30 {
- margin-top: 30px; }
-
-.mt-50 {
- margin-top: 50px; }
-
-.mr-10 {
- margin-right: 10px; }
-
-.mr-20 {
- margin-right: 20px; }
-
-.mr-30 {
- margin-right: 30px; }
-
-.mr-50 {
- margin-right: 50px; }
-
-.mb-10 {
- margin-bottom: 10px; }
-
-.mb-20 {
- margin-bottom: 20px; }
-
-.mb-30 {
- margin-bottom: 30px; }
-
-.mb-50 {
- margin-bottom: 50px; }
-
-.ml-10 {
- margin-left: 10px; }
-
-.ml-20 {
- margin-left: 20px; }
-
-.ml-30 {
- margin-left: 30px; }
-
-.ml-50 {
- margin-left: 50px; }
-
-.my-20 {
- margin-top: 20px !important;
- margin-bottom: 20px !important; }
-
-.mw-100 {
- max-width: 100%; }
-
-.p-0 {
- padding: 0; }
-
-.p-10 {
- padding: 10px; }
-
-.p-20 {
- padding: 20px; }
-
-.p-30 {
- padding: 30px; }
-
-.p-40 {
- padding: 40px; }
-
-.p-50 {
- padding: 50px; }
-
-.p-60 {
- padding: 60px; }
-
-.pl-20 {
- padding-left: 20px; }
-
-.pt-80 {
- padding-top: 80px; }
-
-.py-40 {
- padding-top: 40px;
- padding-bottom: 40px; }
-
-.px-20 {
- padding-left: 20px;
- padding-right: 20px; }
-
-.px-40 {
- padding-left: 40px;
- padding-right: 40px; }
-
-.img-circle {
- border-radius: 50%; }
-
-footer p {
- font-size: 12px !important;
- line-height: 12px !important; }
-
-.absolute {
- position: absolute; }
-
-.relative {
- position: relative; }
-
-.s-48 {
- width: 48px;
- height: 48px; }
-
-.grey-waves {
- background: grey url("/assets/images/sck_bg.png");
- background-repeat: repeat;
- background-size: 65px; }
-
-.new-landing-page h1, .new-landing-page h2, .new-landing-page h3, .new-landing-page h4, .new-landing-page h5, .new-landing-page h6 {
- letter-spacing: 0.0325em;
- font-family: 'Kanit';
- font-weight: 900;
- color: #1C1C1C; }
-
-.new-landing-page h1 {
- font-size: 65px;
- line-height: 65px;
- margin: 0px 0px;
- padding-top: 70px; }
-
-.new-landing-page h2 {
- font-size: 45px;
- line-height: 45px;
- margin: 0px 0px; }
-
-.new-landing-page h3 {
- font-size: 30px;
- line-height: 30px;
- margin-top: 10px;
- margin-bottom: 10px; }
-
-.new-landing-page p {
- color: #1C1C1C;
- font-family: 'Roboto Condensed';
- font-size: 20px;
- line-height: 24px; }
-
-.new-landing-page a {
- text-decoration: none !important; }
-
-.new-landing-page .sc-logo {
- height: 43px;
- position: absolute;
- z-index: 2;
- left: 10%;
- top: 25px; }
-
-.new-landing-page .sc-off-cta-platform {
- position: absolute;
- z-index: 2;
- right: 10%;
- top: 25px; }
-
-.new-landing-page .img-new_sck {
- min-height: 30vw;
- background: blue url("/assets/images/new_sck.jpg") center/cover; }
-
-.new-landing-page .img-sck_edu {
- min-height: 30vw;
- background: blue url("/assets/images/sck_edu.jpg") center/cover; }
-
-.new-landing-page .img-sck_com {
- background: blue url("/assets/images/sck_communities.jpg") center/cover; }
-
-.new-landing-page .img-research {
- background: blue url("/assets/images/sck_research.jpg") center/cover; }
-
-.new-landing-page .img-governm {
- background: blue url("/assets/images/sck_cities.jpg") center/cover; }
-
-.new-landing-page .img-platform {
- min-height: 630px !important;
- background: blue url("/assets/images/sck_platform.jpg") center/cover; }
-
-.new-landing-page .img-docs {
- min-height: 320px !important;
- background: blue url("/assets/images/sck_docs.jpg") top/cover; }
-
-.new-landing-page .tile {
- padding: 60px 40px;
- border-bottom: 3px solid #1C1C1C;
- border-right: 3px solid #1C1C1C; }
-
-.new-landing-page .tile-image {
- min-height: 300px;
- padding: 0 !important; }
-
-.new-landing-page .tile-left {
- border-left: 3px solid #1C1C1C; }
-
-.new-landing-page .tile-top {
- border-top: 3px solid #1C1C1C; }
-
-.new-landing-page .text-funding {
- text-align: left; }
-
-.new-landing-page .video-section {
- background: blue url("/assets/images/landing/smartcitizen-seeed-preorder.jpg") center/cover;
- position: relative;
- height: 90vh;
- width: 100%;
- overflow: hidden; }
-
-.new-landing-page .heading-over-video {
- position: absolute;
- z-index: 1;
- top: 0px;
- height: 100%;
- padding-left: 10%;
- padding-right: 60%; }
- .new-landing-page .heading-over-video img {
- padding-bottom: 40px; }
-
-@media only screen and (max-width: 1010px) {
- .new-landing-page h1 {
- font-size: 42px;
- line-height: 42px; }
- .new-landing-page h2 {
- font-size: 28px;
- line-height: 28px; }
- .new-landing-page h3 {
- font-size: 26px;
- line-height: 26px; }
- .new-landing-page p {
- font-size: 18px;
- line-height: 22px; }
- .new-landing-page .sc-logo {
- left: 5%; }
- .new-landing-page .heading-over-video {
- padding-top: 40px;
- position: absolute;
- padding-left: 5%;
- padding-right: 25%;
- height: 100%; }
- .new-landing-page .heading-over-video img {
- padding-bottom: 00px; } }
-
-@media only screen and (max-width: 750px) {
- .new-landing-page h1 {
- font-size: 30px; }
- .new-landing-page h2 {
- font-size: 25px; }
- .new-landing-page .text-funding {
- text-align: center; } }
-
-@media only screen and (max-width: 600px) {
- .new-landing-page h1 {
- font-size: 28px;
- line-height: 33px;
- padding-bottom: 10px; }
- .new-landing-page h2 {
- font-size: 28px; }
- .new-landing-page p {
- font-size: 19px; }
- .new-landing-page .p-60 {
- padding: 50px 10px !important; }
- .new-landing-page .border-xs-top {
- border-top: 3px solid #1C1C1C;
- border-left: 3px solid #1C1C1C;
- border-right: 3px solid #1C1C1C;
- border-bottom: 3px solid #1C1C1C; }
- .new-landing-page .border-xs-bottom {
- border-top: unset;
- border-left: 3px solid #1C1C1C;
- border-right: 3px solid #1C1C1C;
- border-bottom: 3px solid #1C1C1C; }
- .new-landing-page .img-platform {
- min-height: 356px !important; }
- .new-landing-page .heading-over-video {
- padding-right: 5%; }
- .new-landing-page .sc-off-cta-platform {
- padding: 10px 10px; }
- .new-landing-page .my-20 {
- margin-top: 10px !important;
- margin-bottom: 10px !important; }
- .new-landing-page .img-new_sck {
- background: blue url("/assets/images/sck_edu.jpg") center/cover; }
- .new-landing-page .img-sck_edu {
- background: blue url("/assets/images/new_sck.jpg") center/cover; } }
-
-/**
- * NAVBAR
- */
-.navbar_container a:hover, .navbar_container a:active {
- color: white; }
-
-.navbar_container .md-button.navbar_highlight_button {
- border: 2px solid #FFC100;
- color: #FFC100;
- border-radius: 50px;
- padding-top: .01rem;
- padding-bottom: .01rem;
- transition: color 0.4s cubic-bezier(0.25, 0.8, 0.25, 1), background-color 0.4s cubic-bezier(0.25, 0.8, 0.25, 1); }
- .navbar_container .md-button.navbar_highlight_button:hover, .navbar_container .md-button.navbar_highlight_button:active {
- background-color: #FFC100 !important;
- color: #1C1C1C !important; }
-
-.navbar_container .logo_link {
- text-align: center; }
- .navbar_container .logo_link .logo_icon {
- height: 32px;
- width: 32px; }
-
-.navbar_container .nav_icon {
- margin-right: 4px;
- height: 14px !important;
- width: 14px !important; }
- .navbar_container .nav_icon svg {
- fill: #FFC100; }
-
-.navbar_container .navbar_signup_button .md-button {
- padding: 2px 0px;
- border: 2px solid #FFC100;
- color: #FFC100;
- border-radius: 50px;
- width: 79px; }
-
-.navbar_container .navbar_signup_button .md-button:hover {
- background-color: #FFC100;
- color: #1C1C1C; }
-
-.navbar_avatar_icon {
- width: 38px;
- height: 38px;
- border-radius: 19px; }
-
-.md-open-menu-container.md-active {
- margin-top: 50px !important; }
-
-#search input {
- background: url(../assets/images/search_icon_black.svg);
- background-repeat: no-repeat;
- background-position: 10px 12px;
- background-size: 18px;
- padding: 0px 35px; }
-
-.search_results {
- display: flex;
- align-items: center; }
- .search_results .result_icon {
- flex: 0 0 18px;
- margin-right: 12px;
- height: 18px; }
- .search_results .result_name {
- margin-right: 10px; }
- .search_results .result_location {
- color: #0019FF; }
-
-/*
- CONTAINERS USED ON SHOW KIT STATE
- */
-.over_map {
- min-height: 120px; }
-
-.kit_fixed {
- z-index: 2;
- position: relative; }
-
-.kit_data {
- width: 100%;
- margin-top: 376px; }
- .kit_data section.overlay {
- width: 100%;
- height: 100%;
- position: absolute;
- background-color: white;
- z-index: 3;
- color: #1C1C1C;
- text-align: center;
- font-size: 20px;
- font-weight: lighter; }
- .kit_data section.overlay h1 {
- color: #1C1C1C;
- font-size: 1.8em;
- margin: 0; }
- .kit_data section.overlay p {
- color: #D2D3D5;
- margin: 0.5em; }
- .kit_data .hint {
- background-color: white;
- text-align: center;
- color: #D2D3D5;
- margin-top: 0px;
- z-index: 10;
- position: absolute;
- height: 100%;
- width: 100%; }
- .kit_data .hint > p {
- color: #6E6E6E; }
- .kit_data .hint .md-button {
- margin-top: 20px; }
-
-.kit_data .shadow, .kitTags__section .shadow {
- top: 0px;
- height: 1px;
- background-color: rgba(0, 0, 0, 0.1);
- position: relative;
- box-shadow: 0px -1px 6px 0 rgba(0, 0, 0, 0.65);
- z-index: 2; }
-
-.kit_tags-select-header {
- height: 48px;
- display: flex; }
-
-.kit_tags-header-searchbox {
- border: none;
- border-bottom: 1px solid #ccc;
- padding-left: 12px;
- height: 100%;
- width: 100%; }
-
-/*
-=========================================
-KIT COMMENTS MODULE
-=========================================
-*/
-.kit_comments {
- padding-bottom: 100px; }
-
-/*
- CHART COMPONENT
-*/
-.chart_container {
- /*min-width: 340px;*/
- min-height: 310px;
- /*width: 100%;*/
- /*height: 90%;*/
- position: relative; }
- .chart_container .chart_line {
- fill: none;
- stroke-width: 2px; }
- .chart_container .chart_area {
- opacity: 0.1; }
- .chart_container .axis path, .chart_container .axis line {
- fill: none;
- stroke: #6E6E6E;
- stroke-width: 1;
- shape-rendering: crispEdges;
- display: none; }
- .chart_container .axis {
- font-size: 10px; }
- .chart_container .axis.y_left {
- fill: #D2D3D5; }
- .chart_container .axis.y_right {
- fill: #D2D3D5; }
- .chart_container .axis.x {
- fill: #D2D3D5; }
- .chart_container .grid .tick {
- stroke: #D2D3D5;
- stroke-opacity: 0.6;
- shape-rendering: crispEdges; }
- .chart_container .grid path {
- stroke-width: 0; }
- .chart_container .overlay {
- fill: none;
- pointer-events: all; }
- .chart_container .focus circle {
- fill: #03252d;
- stroke-width: 2px; }
- .chart_container .text_hover_container {
- border: 1px solid #6E6E6E; }
- .chart_container .popup_value {
- fill: #1C1C1C;
- color: #1C1C1C;
- font-size: 18px;
- font-weight: bold; }
- .chart_container .popup_date {
- fill: #6E6E6E;
- font-size: 10px;
- font-weight: bold; }
- .chart_container .popup_unit {
- fill: #D2D3D5;
- margin-left: 10px; }
-
-.stickNav {
- position: fixed;
- top: 0px; }
-
-.overlay-kitinfo {
- background: white;
- height: 100%;
- width: 100%;
- position: absolute;
- z-index: 999; }
-
-/*
- KIT MENU MODULE
- */
-section .kit_menu {
- width: 100%;
- z-index: 5;
- background-color: #ECECEE;
- position: relative;
- /*Show raw Toggle*/
- /* Rounded slider */ }
- section .kit_menu .container {
- width: 100%;
- height: 33px; }
- section .kit_menu .kit_time {
- position: relative;
- text-align: center;
- color: #6E6E6E;
- letter-spacing: 0.5px;
- font-size: 14px;
- text-transform: uppercase; }
- section .kit_menu .kit_user {
- margin-left: 20px;
- line-height: 30px;
- color: #6E6E6E;
- font-weight: bold;
- font-size: 13px; }
- section .kit_menu .kit_user img {
- border-radius: 50%;
- float: left;
- width: 18px;
- height: 18px;
- margin-right: 8px;
- margin-top: 5px; }
- section .kit_menu .kit_user a {
- color: #6E6E6E;
- text-decoration: none; }
- section .kit_menu .kit_name {
- margin-left: 30px;
- line-height: 30px;
- font-size: 13px;
- font-weight: bold;
- color: #6E6E6E; }
- section .kit_menu .kit_name md-icon {
- float: left;
- width: 14px;
- height: 14px;
- margin-right: 8px;
- margin-top: 4px; }
- section .kit_menu .kit_name md-icon svg {
- fill: #FFC100; }
- section .kit_menu .kit_name span {
- vertical-align: middle; }
- section .kit_menu .kit_battery {
- margin-left: 30px;
- line-height: 30px;
- color: #6E6E6E;
- font-size: 12px;
- font-weight: bold; }
- section .kit_menu .kit_battery md-icon {
- margin-bottom: 2px;
- width: 16px;
- margin-right: 8px; }
- section .kit_menu .kit_battery md-icon svg {
- fill: #6E6E6E; }
- section .kit_menu .kit-show-raw {
- margin-right: 6px; }
- section .kit_menu .kit-raw-toggle {
- vertical-align: middle; }
- section .kit_menu .switch {
- vertical-align: middle;
- position: relative;
- display: inline-block;
- width: 30px;
- height: 15px; }
- section .kit_menu .switch input {
- opacity: 0;
- width: 0;
- height: 0; }
- section .kit_menu .slider {
- position: absolute;
- vertical-align: middle;
- cursor: pointer;
- top: 0;
- left: 0;
- right: 0;
- bottom: 0;
- background-color: #ccc;
- transition: .4s; }
- section .kit_menu .slider:before {
- position: absolute;
- content: "";
- height: 11px;
- width: 11px;
- left: 3px;
- bottom: 2px;
- background-color: white;
- transition: .4s; }
- section .kit_menu input:checked + .slider {
- background-color: #FFC100; }
- section .kit_menu input:focus + .slider {
- box-shadow: 0 0 1px #2196F3; }
- section .kit_menu input:checked + .slider:before {
- -webkit-transform: translateX(14px);
- transform: translateX(14px); }
- section .kit_menu .slider.round {
- border-radius: 34px; }
- section .kit_menu .slider.round:before {
- border-radius: 50%; }
- section .kit_menu .kit-show-raw-text {
- vertical-align: middle;
- color: #6E6E6E;
- font-size: 12px;
- font-weight: bold; }
- section .kit_menu .kit_navbar {
- margin-right: 20px; }
- section .kit_menu .kit_navbar md-icon {
- width: 14px; }
- section .kit_menu .kit_navbar md-icon svg {
- fill: #6E6E6E; }
- section .kit_menu .kit_navbar .md-button {
- margin: 0; }
- section .kit_menu .kit_navbar a.md-button:hover, section .kit_menu .kit_navbar .button_active {
- border-bottom: 2px solid #1C1C1C; }
-
-.kit_overview {
- width: 100%;
- height: 96px;
- z-index: 2;
- background-color: #F9F9FB;
- position: relative; }
- .kit_overview .button_scroll {
- margin: 0px;
- background: #ccc; }
- .kit_overview .button_scroll_left {
- left: 0; }
- .kit_overview .button_scroll_right {
- right: 0; }
- .kit_overview .sensors_container {
- width: 90%;
- height: 96px;
- margin: 0 auto;
- overflow-x: hidden;
- overflow-y: hidden; }
- @media (pointer: coarse) {
- .kit_overview .sensors_container {
- overflow-x: auto; } }
- .kit_overview .sensor_container {
- position: relative;
- color: #6E6E6E;
- display: inline-table;
- cursor: pointer;
- text-align: center;
- outline: none; }
- .kit_overview .sensor_container p {
- margin: 0px;
- padding-bottom: 15px;
- font-size: 0.8em;
- font-weight: normal; }
- .kit_overview .sensor_container .clear {
- clear: both; }
- .kit_overview .sensor_container .sensor_right {
- position: absolute;
- right: 15px;
- bottom: 38px;
- max-width: 30px; }
- .kit_overview .sensor_container .sensor_right .sensor_unit {
- margin-top: 8px;
- font-size: 14px; }
- .kit_overview .sensor_container .sensor_right .sensor_arrow {
- margin-top: -15px;
- width: 14px;
- height: 7px; }
- .kit_overview .sensor_container .sensor_right .sensor_arrow.arrow_up svg {
- fill: #00E597; }
- .kit_overview .sensor_container .sensor_right .sensor_arrow.arrow_down svg {
- fill: #FF3D4C; }
- .kit_overview .sensor_container .sensor_right .sensor_arrow.equal svg {
- fill: #55C4F5; }
- .kit_overview .sensor_container .sensor_value {
- margin-top: 10px;
- padding-left: 50px;
- padding-right: 50px; }
- .kit_overview .sensor_container .sensor_value_null {
- opacity: 0.15; }
- .kit_overview .sensor_container .sensor_icon {
- width: 22px;
- height: 22px;
- position: absolute;
- bottom: 45px;
- left: 25px; }
- .kit_overview .sensor_container .sensor_icon svg {
- fill: #6E6E6E; }
- .kit_overview .sensor_container:hover {
- opacity: 0.8; }
- .kit_overview .sensor_container.selected {
- color: #1C1C1C; }
- .kit_overview .sensor_container.selected .sensor_icon svg {
- fill: #1C1C1C; }
-
-/**
- * KIT DETAILS
- *
- */
-.kit_details_notAuth_title {
- margin: 0; }
-
-.kit_details_notAuth_subheader {
- color: #6E6E6E; }
-
-.kit_details_notAuth_button {
- float: right; }
-
-.kit_detailed {
- /*height: 1000px;*/
- z-index: 2;
- background-color: white;
- position: relative; }
- .kit_detailed p.description {
- line-height: inherit;
- font-weight: lighter;
- width: 75%; }
- .kit_detailed .kit_timeOpts {
- margin: 0;
- font-size: 1.3em; }
- .kit_detailed .kit_chart {
- min-height: 340px;
- background-color: white;
- position: relative; }
- .kit_detailed .kit_chart .hint {
- background-color: #1C1C1C;
- opacity: 0.6; }
- .kit_detailed .kit_chart .container {
- padding: 37px;
- height: 100%; }
- .kit_detailed .kit_chart .container .kit_chart_left {
- color: #1C1C1C; }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_select .sensor_icon_selected {
- float: left;
- margin-right: 6px;
- padding-top: 5px; }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_select .sensor_icon_selected svg {
- fill: #1C1C1C; }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_select md-select {
- margin: 0; }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_select md-select .md-select-value .md-text .select_image {
- display: none; }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_description {
- /*//color: $background_blue_light;*/
- /*//height: 80px;*/
- /*//margin-top: 0px; //30;*/
- /*//position: relative;*/ }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_description h6 {
- margin: 0;
- margin-bottom: 3px;
- font-size: 1em; }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_description small {
- line-height: 1em; }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_description a {
- margin-left: 10px;
- text-decoration: none;
- color: #6E6E6E; }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_description .sensor_description_full {
- position: absolute;
- top: -50px;
- margin-top: 10px;
- display: none;
- overflow: visible;
- z-index: 200;
- background-color: rgba(255, 255, 255, 0.8);
- color: #1C1C1C;
- border-radius: 2px;
- line-height: 1.5em;
- padding: 10px;
- box-shadow: 0 1px 5px rgba(0, 0, 0, 0.65); }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_data .sensor_value {
- font-size: 2em;
- font-weight: bold; }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_data .sensor_unit {
- font-size: .8em;
- position: relative;
- top: -8px; }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_data .sensor_data_icon {
- width: 14px;
- height: 14px;
- margin-left: 8px; }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_data_description {
- font-size: 12px;
- display: none;
- position: absolute;
- -webkit-transform: translate(5px, 5px);
- transform: translate(5px, 5px);
- background-color: rgba(255, 255, 255, 0.8);
- color: #1C1C1C;
- border-radius: 2px;
- line-height: 1.5em;
- padding: 4px 6px;
- box-shadow: 0 1px 5px rgba(0, 0, 0, 0.65);
- max-width: 175px; }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_compare {
- font-size: .8em;
- padding-top: 30px;
- margin-top: 10px; }
- .kit_detailed .kit_chart .container .kit_chart_right {
- /*//display: inline-block;*/
- /*//height: 100%;*/ }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation {
- height: 10%;
- font-size: 11px; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation > div {
- margin-left: 20px; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation input {
- margin-left: 10px;
- padding-left: 10px;
- font-size: 12px; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .kit_time md-icon {
- width: 11px;
- height: 11px;
- margin-right: 8px; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .picker_container {
- margin-left: 20px; }
- @media only screen and (max-device-width: 1050px) {
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_navigation {
- justify-content: space-around !important;
- margin: 5px !important; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .picker_container {
- margin: 0 5px 0 5px !important; } }
- @media only screen and (max-device-width: 960px) {
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .picker_container {
- display: flex !important;
- align-content: middle !important;
- margin: 10px 0 0 0 !important;
- width: 100% !important;
- justify-content: space-between !important;
- align-items: center !important; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .word_picker {
- display: inline-block !important;
- width: 100% !important; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move {
- display: flex !important;
- width: 100% !important;
- justify-content: space-between !important;
- margin-top: 15px !important; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_button {
- width: 100% !important; } }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_button {
- margin: 0px -2px 0 -2px;
- width: 40px;
- height: 28px;
- display: inline-flex;
- border: 1px solid #6E6E6E;
- background-color: #F9F9FB; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_right {
- border-top-left-radius: 0px;
- border-bottom-left-radius: 0px; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_right svg {
- fill: #6E6E6E; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_right:hover {
- background-color: #6E6E6E; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_right:hover svg {
- fill: #F9F9FB; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_left svg {
- fill: #6E6E6E; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_left:hover {
- background-color: #6E6E6E; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_left:hover svg {
- fill: #F9F9FB; }
- .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move md-icon {
- width: 5.5px;
- height: 9px;
- position: relative;
- top: 0px; }
- .kit_detailed .kit_details {
- color: #1C1C1C; }
- .kit_detailed .kit_details_content svg {
- fill: #1C1C1C; }
- .kit_detailed .kit_details_content .kit_details_labels {
- margin-top: 30px; }
- .kit_detailed .kit_details_name {
- margin: 5px 0 20px 0; }
- .kit_detailed .kit_details_type > span {
- margin-right: 15px; }
- .kit_detailed .kit_details_type_location svg {
- fill: #6E6E6E; }
- .kit_detailed .kit_details_version span {
- color: #6E6E6E;
- font-weight: bold; }
- .kit_detailed .kit_details_id span {
- color: #6E6E6E;
- font-weight: bold; }
- .kit_detailed .kit_details_description {
- padding-top: 4px; }
- .kit_detailed .kit_details_description_title {
- font-size: 16px;
- margin-bottom: 15px; }
- .kit_detailed .kit_details_description_content {
- font-size: 12px; }
- .kit_detailed .kit_details_manage_buttons {
- margin-bottom: 32px; }
- .kit_detailed .kit_details_button_delete {
- color: #FF3D4C; }
- .kit_detailed .kit_details_key {
- float: left;
- height: 26px; }
- .kit_detailed .md-button.kit_details_key_refreshButton:hover, .kit_detailed .md-button.kit_details_key_refreshButton:focus {
- background-color: #DBDBDB; }
- .kit_detailed .kit_detailed_icon_content {
- width: 14px;
- height: 14px;
- margin-right: 4px;
- position: relative;
- top: -1px; }
- .kit_detailed .kit_detailed_icon_content svg {
- fill: #6E6E6E; }
- .kit_detailed .kit_detailed_icon_title {
- width: 18px;
- height: 18px;
- margin-right: 8px;
- position: relative;
- top: -2px; }
- .kit_detailed .kit_detailed_title_container {
- padding: 20px 0; }
- .kit_detailed .kit_detailed_title {
- font-size: 16px;
- display: inline; }
- .kit_detailed .kit_detailed_content_container {
- margin: 62px 37px 0px; }
- .kit_detailed .kit_owner {
- color: #1C1C1C;
- background-color: #F5F5F5;
- padding-bottom: 40px; }
- .kit_detailed .kit_owner .container {
- width: 95%;
- margin: 0 auto; }
- .kit_detailed .kit_owner .kit_owner_usernameLink {
- text-decoration: none; }
- .kit_detailed .kit_owner .kit_owner_usernameText {
- color: #1C1C1C;
- margin: 50px 0 20px 0; }
- .kit_detailed .kit_owner .kit_owner_location {
- color: #6E6E6E;
- font-weight: bold; }
- .kit_detailed .kit_owner .kit_owner_url a {
- text-decoration: none;
- color: #1C1C1C; }
- .kit_detailed .kit_owner .kit_owner_kits_title {
- margin-bottom: 50px; }
- .kit_detailed .kit_owner .kit_owner_kits_list {
- border: 1px solid #EAECF1;
- height: 82px;
- background-color: white; }
- .kit_detailed .kit_owner .kit_owner_kits_list_avatar {
- float: left;
- width: 44px;
- height: 44px;
- margin-right: 3%;
- margin-top: 9px; }
- .kit_detailed .kit_owner .kit_owner_kits_list_content {
- float: left;
- margin-top: 5px; }
- .kit_detailed .kit_owner .kit_owner_kits_list_data {
- font-size: 16px; }
- .kit_detailed .kit_owner .kit_owner_kits_list_data span {
- margin-right: 15px; }
- .kit_detailed .kit_owner .kit_owner_kits_list_state {
- display: inline;
- font-size: 12px;
- border-radius: 20px;
- padding: 6px 11px;
- margin: 15px; }
- .kit_detailed .kit_owner .kit_owner_kits_list_never_published {
- background-color: #FFC100; }
- .kit_detailed .kit_owner .kit_owner_kits_list_not_configured {
- background-color: #FF3D4C; }
- .kit_detailed .kit_owner .kit_owner_kits_list_config {
- float: right;
- margin-top: 13px; }
- .kit_detailed .kit_owner .kit_owner_kits_list_config_icon {
- width: 22px;
- height: 22px; }
-
-.sensor_value {
- font-size: 34px;
- font-family: Kanit;
- font-weight: 700; }
-
-/**/
-/* [!] Leave animation (ng-leave ) are disabled
-because they conflict with enter animations */
-.sensor_animation.ng-enter {
- transition: 1s ease-out all; }
-
-.sensor_animation.ng-enter {
- -webkit-transform: scale(1.05);
- transform: scale(1.05); }
-
-.sensor_animation.ng-enter-active {
- -webkit-transform: scale(1);
- transform: scale(1); }
-
-/* .sensor_animation.ng-leave{
- transition:1s ease all;
-}
-.sensor_animation.ng-leave{
- opacity: 0;
-}
-.sensor_animation.ng-leave-active{
- opacity: 1;
-} */
-/* Time animation currently not in use */
-.time_animation.ng-enter {
- transition: 1s ease-out all; }
-
-.time_animation.ng-enter {
- -webkit-transform: scale(1.05);
- transform: scale(1.05); }
-
-.time_animation.ng-enter-active {
- -webkit-transform: scale(1);
- transform: scale(1); }
-
-/* .time_animation.ng-leave {
- transition:1s ease all;
-}
-.time_animation.ng-leave {
- opacity: 0;
-}
-.time_animation.ng-leave-active {
- opacity: 1;
-} */
-.bat_animation.ng-enter {
- transition: 1s ease-out all; }
-
-.bat_animation.ng-enter {
- -webkit-transform: scale(1.05);
- transform: scale(1.05); }
-
-.bat_animation.ng-enter-active {
- -webkit-transform: scale(1);
- transform: scale(1); }
-
-/* .bat_animation.ng-leave {
- transition:2s ease all;
-}
-.bat_animation.ng-leave {
- opacity: 0;
-}
-.bat_animation.ng-leave-active {
- opacity: 1;
-} */
-/*
- MODAL STORE
-*/
-.store_itemDescription {
- margin-left: 20px; }
- .store_itemDescription ul {
- list-style-image: url("../assets/images/check_circle.svg");
- list-style-position: inherit;
- font-size: .8em;
- line-height: 1.8em; }
-
-.csv_file_item {
- display: flex;
- align-items: center; }
- .csv_file_item md-checkbox {
- margin-bottom: 0px; }
- .csv_file_item span {
- margin-right: 20px; }
-
-.list-shadow {
- box-shadow: 0px 1px 3px 0px rgba(0, 0, 0, 0.2); }
- .list-shadow md-list-item {
- border-bottom: solid 1px #ECECEE; }
- .list-shadow md-list-item:last-child {
- border-bottom: none; }
-
-.label-grey {
- background-color: #D2D3D5; }
-
-.title-timeline {
- font-size: 30px; }
-
-/* This is temporary */
-.upload-csv .container {
- margin-right: auto;
- margin-left: auto; }
-
-.upload-csv .csv_content {
- margin-top: 240px;
- min-height: 50vh; }
-
-@media (min-width: 768px) {
- .upload-csv .container {
- width: 750px; } }
-
-@media (min-width: 992px) {
- .upload-csv .container {
- width: 970px; } }
-
-@media (min-width: 1200px) {
- .upload-csv .container {
- width: 1170px; } }
-
-/**
- * BACKDROP COMPONENT
- */
-md-content.md-opaque {
- background-color: rgba(30, 30, 30, 0.6); }
-
-md-content.md-mainBackdrop {
- display: flex;
- align-items: center;
- justify-content: center;
- width: 100vw;
- height: 100vh; }
- md-content.md-mainBackdrop .backdrop_icon {
- width: 101px;
- height: 101px; }
-
-md-content.md-stateChangeBackdrop {
- z-index: 999;
- position: absolute;
- background-color: transparent;
- width: 100vw;
- height: 100vh; }
-
-.backdrop_icon svg {
- overflow: visible;
- fill: #6E6E6E; }
-
-.backdrop_icon .sc-circle-arrow {
- -webkit-transform-origin: 50% 53.8%;
- transform-origin: 50% 53.8%;
- -webkit-animation: spin 3s linear infinite;
- animation: spin 3s linear infinite; }
-
-@-webkit-keyframes spin {
- 100% {
- -webkit-transform: rotate(360deg); } }
-
-@keyframes spin {
- 100% {
- -webkit-transform: rotate(360deg);
- transform: rotate(360deg); } }
-
-section .stickMenu {
- position: absolute !important; }
-
-.stickNav {
- position: fixed;
- top: 0px;
- border-bottom: 1px solid #666; }
-
-/**
- * MOZILLA-SPECIFIC CSS
- */
-@-moz-document url-prefix() {
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_select .sensor_icon_selected {
- transform: translateY(-8px); }
- .kit_detailed .kit_chart .container .kit_chart_left .sensor_compare md-select {
- transform: translateY(8px);
- margin: 0; }
- body .kit_detailed .kit_chart .container .kit_chart_left .sensor_data {
- margin: 5% 0 4% 0; }
- /* section.map {
- z-index: 0;
- } */ }
-
-/**
- * INPUT CSS
- */
-html {
- font-family: 'Roboto Condensed';
- font-size: 18px; }
-
-body {
- background-color: rgba(0, 0, 0, 0);
- color: #1C1C1C; }
-
-a.about {
- cursor: pointer;
- text-decoration: none;
- color: #0019FF; }
-
-li.policy-toc {
- padding: 3px; }
- li.policy-toc a {
- text-decoration: none; }
-
-ul#policy-toc {
- padding-bottom: 0px !important;
- padding-bottom: 0px !important; }
-
-.policy-toc#header {
- list-style: none; }
-
-.icon_label {
- width: 14px;
- height: 14px;
- margin-right: 4px;
- position: relative;
- top: -2px; }
- .icon_label svg {
- fill: #6E6E6E; }
-
-/**
- * TYPOGRAPHY
- */
-h1 {
- font-size: 2.2em;
- font-weight: normal; }
-
-h2 {
- font-size: 1.7em;
- font-weight: normal; }
-
-h3 {
- font-size: 1.4em;
- font-weight: normal; }
-
-h4 {
- font-size: 1.2em;
- font-weight: normal; }
-
-p {
- line-height: 1.5; }
-
-td {
- padding: 5px; }
-
-input {
- font-family: 'Roboto Condensed'; }
-
-.label, .tag, .state {
- text-transform: uppercase;
- font-size: 12px;
- line-height: 24px;
- border-radius: 20px;
- padding: 3px 8px;
- margin-right: 8px; }
- .label md-icon, .tag md-icon, .state md-icon {
- margin-left: 4px;
- height: 11px;
- width: 11px;
- margin-top: -3px;
- cursor: pointer; }
- .label[clickable], .tag[clickable], .state[clickable] {
- cursor: pointer; }
-
-.label {
- background-color: #FFD44D;
- color: #1C1C1C; }
- .label md-icon svg path {
- fill: #D2D3D5; }
-
-.tag {
- background-color: #FFECB3;
- color: #1C1C1C; }
- .tag md-icon svg path {
- fill: #6E6E6E; }
-
-.leaflet-container .tag, .leaflet-container .label,
-.leaflet-container .state {
- font-size: 1em; }
-
-.leaflet-top {
- top: 93px; }
-
-section.info h1 {
- font-size: 1.3em; }
-
-a {
- color: #1C1C1C; }
- a:hover, a:active {
- text-decoration: none;
- color: #0019FF; }
-
-a.footer-link {
- color: #FFC100;
- text-decoration: none;
- transition: color 0.4s cubic-bezier(0.25, 0.8, 0.25, 1); }
- a.footer-link:hover, a.footer-link:active {
- text-decoration: none;
- color: #0019FF; }
-
-/*
- Overwrite global material design, allow icons to be smaller
-*/
-md-toolbar {
- background-color: #1C1C1C !important;
- z-index: 10;
- color: white !important; }
- md-toolbar md-icon {
- fill: white !important; }
-
-md-icon {
- min-width: unset !important; }
-
-md-select md-select-value {
- border-bottom-color: #cccccc75 !important; }
-
-a.md-button h4, a.md-button .h4 {
- color: #0019FF;
- margin: 0; }
-
-a.warn,
-button.warn {
- color: #FF3D4C; }
-
-.bold {
- font-weight: bold; }
-
-div[api-key] {
- display: flex; }
- div[api-key] .api_key_number {
- background-color: #F2F2F2;
- word-break: break-all;
- width: 420px; }
- div[api-key] .api_key_refresh_button {
- background-color: #DBDBDB;
- margin: 0; }
- div[api-key] .api_key_refresh_button md-icon {
- padding: 10px;
- height: 15px;
- width: 15px; }
- div[api-key] md-icon {
- opacity: .5; }
-
-.info_overlay h2 {
- color: white; }
-
-.info_overlay p {
- margin: 0;
- font-size: 20px;
- font-weight: 300;
- color: #D2D3D5; }
-
-.dark h1, .dark h2 {
- color: white; }
-
-.dark h1, .dark h2 {
- margin: 0; }
-
-.dark h3, .dark h4, .dark .h4, .dark a {
- color: #D2D3D5; }
-
-/**
- * ANGULAR MATERIAL OVERRIDING
- */
-md-select-menu {
- background-color: white; }
-
-md-select,
-md-select-menu {
- text-transform: uppercase; }
-
-md-select:not([disabled]):focus .md-select-value {
- border-bottom-color: #6B868D; }
-
-.md-button.md-primary.md-flat:not([disabled]):hover {
- background-color: #0019FF;
- color: white; }
-
-.md-button.md-primary.md-flat:not([disabled]):focus {
- background-color: #0019FF;
- color: white; }
-
-.md-button.fillwidth {
- width: 100% !important; }
-
-md-input-container:not(.md-input-invalid).md-input-focused label {
- color: #0019FF; }
-
-md-input-container:not(.md-input-invalid).md-input-focused .md-input {
- border-color: rgba(0, 0, 0, 0.12); }
-
-/**
- * ALERT COMPONENT
- */
-.md-toast-content {
- box-shadow: unset !important;
- background-color: unset !important;
- color: black !important;
- min-width: 100%; }
-
-md-toast {
- bottom: unset !important;
- color: black;
- opacity: 0.95;
- padding: 0;
- position: fixed;
- top: 64px !important;
- min-width: 100%;
- z-index: 15; }
- md-toast .alert_typeIcon {
- width: 16px;
- height: 16px;
- margin-right: 16px; }
- md-toast .alert_closeIcon {
- width: 10.5px;
- height: 10.5px;
- margin: 0 auto; }
- md-toast .alert_button {
- padding: 0px 12px;
- margin-left: 5px;
- color: black !important;
- background-color: #FFC100;
- border-radius: 20px;
- font-size: 12px; }
-
-md-toast.red {
- background-color: #FF3D4C; }
-
-md-toast.green {
- background-color: #4DBD4E; }
-
-md-toast.blue {
- background-color: #0019FF; }
-
-md-toast.yellow {
- background-color: #FFC100; }
-
-md-checkbox.md-checked .md-icon {
- background-color: #0019FF; }
-
-md-checkbox .md-label {
- pointer-events: auto; }
-
-md-checkbox .md-label span {
- pointer-events: none;
- -webkit-user-select: none;
- -moz-user-select: none;
- -ms-user-select: none;
- user-select: none; }
-
-/**
- * SPINNER/PROGRESS BAR COMPONENTS
- */
-md-progress-linear.md-bar.green {
- background-color: #00E597; }
-
-md-progress-linear.green .md-container {
- background-color: rgba(0, 229, 151, 0); }
-
-md-progress-linear.green .md-bar {
- background-color: #00E597; }
-
-md-progress-linear.md-bar {
- background-color: #0019FF; }
-
-md-progress-linear.md-hue-3 .md-container {
- background-color: rgba(0, 25, 255, 0.05); }
-
-md-progress-linear.md-hue-3 .md-bar {
- background-color: #0019FF; }
-
-.kit_spinner {
- z-index: 200;
- position: absolute;
- left: 0;
- bottom: 6px; }
-
-.chart_spinner {
- position: absolute;
- top: 200px;
- left: 57%; }
-
-/**
- * COOKIES POLICY COMPONENT
- */
-.cookies-policy_container {
- background-color: #3c3c3c;
- color: #FFC100;
- position: fixed;
- bottom: 0;
- padding: 5px;
- left: 0;
- width: 100%;
- z-index: 999;
- text-align: center;
- line-height: 4vh; }
- .cookies-policy_container a {
- color: rgba(255, 193, 0, 0.6); }
-
-.md-noDataBackdrop {
- width: 100%;
- height: 381px;
- position: absolute;
- background-color: white; }
- .md-noDataBackdrop .block {
- margin: 0 auto; }
- .md-noDataBackdrop .title, .md-noDataBackdrop .message {
- text-align: center; }
- .md-noDataBackdrop .title {
- color: #6E6E6E;
- font-weight: 400;
- margin-bottom: 0; }
- .md-noDataBackdrop .message {
- margin: 0;
- font-size: 20px;
- font-weight: 300; }
- .md-noDataBackdrop a {
- color: #0019FF;
- text-decoration: none; }
-
-/**
- * MAP
- */
-.map_state {
- position: fixed;
- z-index: 1; }
-
-section.map {
- background-color: white;
- text-align: center;
- width: 100%;
- height: 100%;
- color: black;
- z-index: 1;
- position: fixed;
- top: 64px; }
- @media all and (max-height: 1200px) and (min-height: 1000px) {
- section.map {
- height: 950px; } }
- @media all and (max-height: 1000px) and (min-height: 800px) {
- section.map {
- height: 700px; } }
- @media all and (max-height: 800px) and (min-height: 600px) {
- section.map {
- height: 550px; } }
- @media all and (max-height: 600px) and (min-height: 400px) {
- section.map {
- height: 300px; } }
- section.map h1 {
- font-size: 3em; }
-
-/*
- CORE CHIP COMPONENT
- */
-.chips {
- display: inline-block; }
- .chips div.chips_row {
- height: 40.5px; }
- .chips .label, .chips .tag {
- box-shadow: 0 1px 5px rgba(0, 0, 0, 0.65); }
-
-.chip {
- padding: 0px 10px 0px 14px;
- margin: auto 5px;
- border-radius: 20px;
- height: 30px;
- text-transform: uppercase;
- font-size: 13px;
- line-height: 29px; }
- .chip .chip_name {
- margin-left: 3px;
- margin-right: 7px; }
- .chip .chip_icon {
- border: 0;
- background-color: inherit; }
- .chip .chip_icon md-icon {
- width: 8px;
- height: 8px; }
- .chip.map_filter {
- background-color: white;
- box-shadow: 0 1px 5px rgba(0, 0, 0, 0.65); }
- .chip.kit_tag {
- background-color: #D2D3D5;
- float: left;
- margin: 5px; }
-
-/*
- MAP FILTERS
-*/
-.map_legend {
- position: absolute;
- top: 10px;
- margin-left: 10px;
- display: flex; }
- .map_legend .chips {
- align-self: flex-end;
- padding-bottom: 5px;
- padding-left: 5px;
- display: flex; }
- .map_legend .map_legend__filtersContainer {
- display: flex;
- flex-direction: column;
- justify-content: space-between;
- width: 124px;
- height: 81px;
- border-radius: 4px;
- overflow: hidden;
- box-shadow: 0 1px 5px rgba(0, 0, 0, 0.65); }
- .map_legend .map_legend__filtersContainer svg, .map_legend .map_legend__filtersContainer p {
- fill: #1C1C1C; }
- .map_legend .map_legend__filtersContainer .map_legend__filtersRow {
- cursor: pointer;
- display: flex;
- align-items: center;
- padding-left: 5px;
- background-color: #fff; }
- .map_legend .map_legend__filtersContainer .map_legend__filtersRow:not(:last-child) {
- border-bottom: 1px solid gray; }
- .map_legend .map_legend__filtersContainer .map_legend__filtersRow .map_filter_button {
- box-shadow: none; }
- .map_legend .map_legend__filtersContainer .map_legend__filtersRow .map_filter_button p {
- margin: 0; }
-
-.map_filter_button {
- background-color: white;
- margin-right: 15px;
- width: 26px;
- height: 26px;
- box-shadow: 0 1px 5px rgba(0, 0, 0, 0.65); }
- .map_filter_button md-icon {
- width: 18px;
- height: 12px; }
-
-.md-button.map_filter_button:hover, .md-button.map_filter_button:focus {
- background-color: white; }
-
-/*
-=========================================
-KIT TAGS MODULE
-=========================================
-*/
-.kitTags__section {
- margin-top: 376px; }
- .kitTags__section .kitTags__listContainer {
- background-color: #ECECEE;
- padding: 20px; }
- .kitTags__section .kitTags__container {
- padding-bottom: 20px; }
- .kitTags__section .kitTags__container .kitTags__textContainer {
- display: flex;
- justify-content: space-between;
- margin-left: 20px;
- margin-right: 20px; }
- .kitTags__section .kitTags__container .kitTags__textContainer .kitTags__textElement {
- flex-basis: 40%;
- flex-grow: 0;
- min-width: 300px; }
-
-/*
- CORE FORM COMPONENT
- */
-.form_container {
- font-size: 16px;
- position: relative;
- border-radius: 0px 0px 3px 3px; }
-
-.form_contentContainer {
- width: 92%;
- margin: 0 auto; }
-
-.form_field {
- position: relative; }
-
-.form_errors {
- position: absolute;
- bottom: 15%;
- left: 5%;
- color: red;
- font-size: 0.7em;
- line-height: 1; }
-
-.form_closeIcon {
- float: right;
- width: 14px;
- height: 14px;
- margin: 12px; }
- .form_closeIcon md-icon {
- width: 14px;
- height: 14px;
- position: absolute;
- top: 0px;
- left: 0px; }
-
-.md-primary.form_button {
- color: white;
- background-color: #0019FF;
- height: 64px;
- bottom: 0px;
- left: 0px;
- border-radius: 0px 0px 2px 2px; }
-
-.form_title {
- color: #0019FF;
- font-weight: normal; }
-
-.form_messageHeader {
- font-size: 1.5em;
- color: #6E6E6E;
- font-weight: bold;
- line-height: 0;
- margin-top: 50px; }
-
-.form_messageSubheader {
- font-size: 0.875em;
- color: rgba(0, 98, 123, 0.54);
- line-height: 0.5; }
-
-.form_messageDescription {
- font-size: 12px;
- margin-bottom: 0; }
-
-/*
- CORE LARGE DEVICES FORM
-*/
-@media (min-width: 501px) {
- .form_container {
- width: 340px;
- font-size: 16px; }
- .form_messageContainer {
- margin-bottom: 20px; }
- .md-primary.form_button {
- width: 340px;
- display: block;
- position: relative;
- margin: 0 auto;
- font-size: 1.5em; } }
-
-/*
- CORE MOBILE DEVICES FORM
-*/
-@media (max-width: 500px) {
- .form_container {
- width: 80%;
- font-size: 12px; }
- .md-primary.form_button {
- width: 100%;
- position: absolute;
- font-size: 1.2em; } }
-
-@media only screen and (min-device-width: 320px) and (max-device-width: 480px) and (-webkit-device-pixel-ratio: 2) and (device-aspect-ratio: 2 / 3) {
- .recovery_container {
- margin-top: 40px; }
- .form_field {
- padding: 5% 0; } }
-
-/*
- RECOVERY FORM
-*/
-.recovery_container {
- position: relative;
- margin: 0px auto;
- margin-bottom: 100px; }
-
-.formRecovery_field {
- padding: 10% 0; }
-
-@media (min-width: 501px) {
- .recovery_container {
- margin-top: 80px; }
- .formRecovery_errors {
- bottom: 15%;
- left: 5%; } }
-
-@media (max-width: 500px) {
- .recovery_container {
- margin-top: 20%; }
- .formRecovery_errors {
- bottom: 23%;
- left: 5%; } }
-
-@media only screen and (min-device-width: 320px) and (max-device-width: 480px) and (-webkit-device-pixel-ratio: 2) and (device-aspect-ratio: 2 / 3) and (max-width: 500px) {
- .formRecovery_field {
- padding: 2% 0; }
- .formRecovery_errors {
- bottom: 29%; } }
-
-/*
- RESET FORM
-*/
-.formReset_field {
- padding: 7% 0; }
-
-@media (min-width: 501px) {
- .formReset_errors {
- bottom: 5%;
- left: 5%; } }
-
-@media (max-width: 500px) {
- .formReset_errors {
- bottom: 18%;
- left: 1%; } }
-
-@media only screen and (min-device-width: 320px) and (max-device-width: 480px) and (-webkit-device-pixel-ratio: 2) and (device-aspect-ratio: 2 / 3) and (max-width: 500px) {
- .formReset_errors {
- top: 65%; }
- .formReset_field {
- padding: 3% 0; } }
-
-/**
- * MOVE DOWN EFFECT ON SCROLL
- */
-.move_down {
- -webkit-transform: translateY(32px);
- transform: translateY(32px); }
-
-/*
- MARKER AND POPUP MARKER
-*/
-.markerSmartCitizenNormal {
- border: 1px solid #FFC100;
- border-radius: 20px;
- box-shadow: 0px 2px 4px 0px rgba(0, 0, 0, 0.3); }
-
-.markerSmartCitizenOnline {
- border: 2px solid #FFC100;
- border-radius: 20px;
- box-shadow: 0px 2px 4px 0px rgba(0, 0, 0, 0.3);
- background-color: rgba(255, 193, 0, 0.5); }
-
-.markerExperimentalNormal {
- border: 2px solid #D2D3D5;
- border-radius: 20px;
- box-shadow: 0px 2px 4px 0px rgba(0, 0, 0, 0.3);
- background-color: #D2D3D5; }
-
-/* @keyframes pulse {
- 0% {
- opacity: 0.2;
- }
- 50% {
- opacity: 1;
- }
- 100% {
- opacity: 0.2;
- }
-}
-@-webkit-keyframes pulse {
- 0% {
- opacity: 0.2;
- }
- 50% {
- opacity: 1;
- }
- 100% {
- opacity: 0.2;
- }
-}
-@-moz-keyframes pulse {
- 0% {
- opacity: 0.2;
- }
- 50% {
- opacity: 1;
- }
- 100% {
- opacity: 0.2;
- }
-}
-@-o-keyframes pulse {
- 0% {
- opacity: 0.2;
- }
- 50% {
- opacity: 1;
- }
- 100% {
- opacity: 0.2;
- }
-} */
-.markerSmartCitizenOffline {
- border: 2px solid rgba(0, 0, 0, 0.2);
- border-radius: 20px;
- box-shadow: 0px 2px 4px 0px rgba(0, 0, 0, 0.3);
- background-color: rgba(0, 0, 0, 0.2); }
-
-/* .marker_blink {
- -webkit-animation: pulse 2s linear infinite;
- -moz-animation: pulse 2s linear infinite;
- -ms-animation: pulse 2s linear infinite;
- animation: pulse 2s linear infinite;
-} */
-.popup {
- width: 101%;
- height: 135px;
- color: #1C1C1C;
- font-family: 'Roboto Condensed'; }
-
-.popup_top {
- height: 72px;
- border-radius: 2px 2px 0 0;
- background-color: #ECECEE;
- padding-top: 5px; }
- .popup_top a {
- color: #1C1C1C; }
- .popup_top a:hover {
- color: #6E6E6E; }
- .popup_top .popup_name {
- padding: 0 0 1px 8px;
- font-size: 16px;
- font-weight: bold; }
- .popup_top .popup_type {
- font-size: 14px;
- line-height: 1em;
- padding: 1px 0 3px 8px; }
- .popup_top .popup_time {
- font-size: 11px;
- padding: 2px 0 0 8px; }
- .popup_top .popup_time md-icon {
- width: 10px;
- height: 10px;
- margin-right: 5px; }
-
-.popup_bottom {
- height: 68px; }
- .popup_bottom .popup_location {
- padding: 5px 0 0 8px;
- font-size: 14px; }
- .popup_bottom .popup_location md-icon {
- width: 8.4px;
- height: 12px;
- margin-right: 4px; }
- .popup_bottom .popup_labels {
- padding: 8px 0 0 8px;
- font-size: 12px; }
- .popup_bottom .popup_labels tag, .popup_bottom .popup_labels span {
- margin-bottom: 8px;
- float: left; }
-
-.popup_icon svg:nth-child(2) {
- width: 0;
- height: 0; }
-
-/* Kit type colors kitUtils.service.js :: classify()*/
-.sck {
- background-color: #FFC100; }
-
-.unknown {
- background-color: grey; }
-
-/**
- * OVERRIDE LEAFLET
- */
-.leaflet-left {
- left: unset !important;
- right: 11px;
- top: 15px !important; }
-
-.leaflet-popup-content-wrapper {
- border-radius: 4px;
- width: auto !important;
- overflow: hidden; }
-
-.leaflet-popup-content {
- margin: 0;
- min-width: 245px;
- position: relative;
- top: -1px;
- left: -1px; }
- .leaflet-popup-content p {
- margin: 0; }
-
-.leaflet-popup-close-button {
- display: none; }
-
-.leaflet-overlay-pane .leaflet-zoom-animated path {
- stroke: grey; }
-
-.leaflet-top.leaflet-left {
- z-index: 1; }
-
-.leaflet-bottom.leaflet-right {
- z-index: 1; }
-
-.marker-cluster {
- box-shadow: 0px 2px 4px 0px rgba(0, 0, 0, 0.2); }
- .marker-cluster div {
- font-family: 'Roboto Condensed';
- font-size: 14px; }
-
-.marker-cluster-small, .marker-cluster-medium, .marker-cluster-large {
- border-radius: 50%;
- border: 2px solid rgba(255, 193, 0, 0.8);
- background-color: rgba(255, 193, 0, 0.3); }
- .marker-cluster-small div, .marker-cluster-medium div, .marker-cluster-large div {
- background-color: rgba(0, 255, 168, 0); }
-
-/*
- PICKER COMPONENT
-*/
-.picker {
- color: #1C1C1C; }
-
-.picker__weekday {
- color: #6E6E6E; }
-
-.picker__nav--prev::before {
- border-right: 0.75em solid #6E6E6E; }
-
-.picker__nav--next::before {
- border-left: 0.75em solid #6E6E6E; }
-
-.picker__button--today::before {
- border-top: 0.66em solid #0019FF; }
-
-.picker__holder.picker_container {
- background: none;
- transition: none; }
-
-.picker__button--clear::before {
- border-top: 3px solid #FF3D4C; }
-
-.picker__button--close {
- color: #1C1C1C; }
-
-.picker__button--today, .picker__button--clear {
- color: #1C1C1C; }
-
-.picker__day--highlighted, .picker__day--selected, .picker__day--selected:hover, .picker--focused .picker__day--selected {
- background: #FFC100;
- background-color: #FFC100;
- border-color: #FFC100;
- color: #1C1C1C; }
-
-.picker__day--infocus:hover, .picker__nav--prev:hover, .picker__nav--next:hover, .picker__button--today:hover, .picker__button--clear:hover, .picker__button--close:hover {
- background: #FFECB3;
- border-color: #FFECB3; }
-
-.date_picker {
- height: 24px;
- border-radius: 2px;
- border: 1px solid #6E6E6E;
- color: #6E6E6E;
- background-color: transparent; }
-
-.select_image {
- margin-right: 10px; }
-
-/**
- * DROPDOWN MENU
- * Override style for dropdown menu from angular-dropdowns dependency
- */
-.wrap-dd-menu .dropdown li a {
- color: #6E6E6E; }
-
-.kit_menu .wrap-dd-menu {
- width: 30px;
- display: inline-table;
- padding: 0 0; }
-
-.kit_menu .wrap-dd-menu .dropdown {
- top: 70%;
- line-height: 1em;
- left: -25px;
- right: -25px; }
- .kit_menu .wrap-dd-menu .dropdown li a {
- text-align: center;
- border-bottom: 0px; }
-
-.kitList_parent .wrap-dd-menu {
- width: 50px;
- display: inline-block; }
-
-.kitList_parent .wrap-dd-menu .dropdown {
- top: 70%;
- line-height: 1em;
- left: -25px;
- right: -25px;
- border-radius: 0px; }
- .kitList_parent .wrap-dd-menu .dropdown li a {
- text-align: center;
- border-bottom: 0px; }
-
-.kit_owner_dropdown_button {
- display: block;
- margin: 0 auto; }
-
-/**
- * KIT LIST
- *
- */
-.kitList {
- margin: 0px;
- color: #D2D3D5;
- border-top: 1px solid #EAECF1;
- border-right: 1px solid #EAECF1;
- border-left: 1px solid #EAECF1;
- background-color: white;
- border-radius: 0 !important;
- text-transform: none;
- padding: 15px; }
- .kitList h1, .kitList h2, .kitList h3, .kitList h4 {
- line-height: inherit; }
- .kitList p {
- line-height: 0.4em; }
- .kitList .wrap-dd-menu {
- width: 50px;
- display: inline-block; }
- .kitList .wrap-dd-menu .dropdown {
- top: 70%;
- line-height: 1em;
- left: -25px;
- right: -25px;
- border-radius: 0px; }
- .kitList .wrap-dd-menu .dropdown li a {
- text-align: center;
- border-bottom: 0px; }
- .kitList .chips {
- position: absolute;
- right: 130px; }
-
-.kitList_avatar {
- width: 44px;
- height: 44px;
- border-radius: 22px; }
-
-.kitList_name {
- margin-bottom: 5px;
- font-size: 18px;
- color: #0019FF; }
-
-.kitList_right {
- position: absolute;
- right: 0; }
- .kitList_right .chip .chip_name {
- text-transform: uppercase; }
-
-.kitList_state {
- color: white;
- text-transform: uppercase; }
-
-.kitList_state_has_published {
- color: #00E597; }
-
-.kitList_state_never_published {
- color: #FFC100; }
-
-.kitList_state_not_configured {
- color: #F43D4D; }
-
-.kitList_dropdownButton {
- display: block;
- margin: 0 auto; }
- .kitList_dropdownButton img {
- position: relative; }
-
-.kitList_noKits {
- line-height: 5; }
-
-.dropdown-item-button {
- border-radius: 5px;
- text-align: center;
- width: 100%; }
-
-.dropdown-item-button:hover {
- background-color: #FFC100 !important; }
-
-md-card.kit-list-item {
- border-radius: 10px; }
-
-/**
- * USER PROFILE STATE
- * most styles are re-used on my profile state
- */
-/* profile header */
-.profile_header {
- background-color: #2E2E2E;
- color: #D2D3D5;
- height: 192px;
- margin-top: 64px; }
-
-.profile_header_avatar {
- width: 100px;
- height: 100px;
- border-radius: 50px;
- margin-right: 29px;
- margin-left: 27px; }
-
-.profile_header_content > div {
- margin: 5px 0; }
-
-.profile_header_content svg {
- fill: #ECECEE; }
-
-.profile_header_content_avatar {
- width: 12px;
- height: 12px;
- margin-right: 6px;
- position: relative;
- top: -1px; }
-
-.profile_content {
- min-height: 700px;
- position: relative; }
-
-/* profile sidebar */
-.profile_sidebar {
- background-color: #F5F5F5;
- min-width: 200px; }
-
-.profile_sidebar_title {
- color: #1C1C1C;
- font-weight: bold; }
-
-.profile_sidebar_description {
- font-size: 14px; }
-
-.profile_sidebar_button {
- height: 64px;
- color: #0019FF; }
- .profile_sidebar_button svg {
- fill: #0019FF; }
-
-.profile_sidebar_avatar {
- width: 12px;
- height: 12px;
- margin-right: 8px;
- position: relative;
- top: -2px; }
-
-/* profile main content */
-.profile_content_main_top {
- margin: 0 0 42px 20px; }
-
-/**
- * MY PROFILE STATE
- */
-.myProfile_state md-ink-bar {
- background-color: #0019FF; }
-
-.myProfile_state md-tab md-tab-label {
- overflow: visible; }
-
-.myProfile_state md-tabs md-tab {
- color: #D2D3D5; }
-
-.myProfile_state md-tabs md-tab.active {
- color: white; }
-
-.myProfile_state md-tabs md-tabs-ink-bar {
- color: #0019FF;
- background-color: #0019FF; }
-
-md-tabs-canvas {
- background-color: #2E2E2E; }
-
-.myProfile_header {
- height: 188px;
- position: relative; }
-
-.myProfile_header_container {
- padding-top: 51px;
- margin-left: 46px; }
-
-.myProfile_header_avatar {
- margin-left: 0; }
-
-.myProfile_tabs_parent md-tabs-wrapper {
- background-color: #1C1C1C; }
-
-.myProfile_tab_icon {
- width: 16px;
- height: 16px;
- margin-right: 8px;
- position: relative;
- top: -1px; }
- .myProfile_tab_icon svg {
- fill: white; }
-
-.myProfile_sidebar_button {
- padding-left: 28px; }
-
-.myProfile_content_form {
- margin-left: 113px;
- max-width: 500px; }
-
-.myProfile_content_form_input {
- margin: 16px 0;
- position: relative; }
-
-.countryInput_container md-autocomplete {
- background-color: transparent; }
-
-.countryInput_container md-autocomplete-wrap {
- box-shadow: none !important; }
-
-.countryInput_container input {
- font-size: 100%; }
-
-.countryInput_container label {
- position: relative;
- bottom: 55px; }
-
-.myProfile_form_avatar {
- margin-bottom: 30px; }
-
-.myProfile_form_avatarImage {
- width: 64px;
- height: 64px;
- border-radius: 32px;
- margin-right: 7%; }
-
-.myProfile_apiKey_text {
- color: #D2D3D5;
- margin-right: 5px;
- width: 120px; }
-
-.myProfile_apiKey_number {
- color: #9D9D9E;
- font-weight: bold;
- background-color: #F2F2F2;
- padding: 5px 10px; }
-
-.myProfile_apiKey_refreshButton {
- background-color: #D2D3D5;
- height: 29px; }
- .myProfile_apiKey_refreshButton md-icon {
- height: 15px;
- width: 15px;
- display: block;
- margin: 3px auto; }
-
-.md-button.myProfile_apiKey_refreshButton {
- border-radius: 0 4px 4px 0; }
-
-.md-button.myProfile_apiKey_refreshButton:active, .md-button.myProfile_apiKey_refreshButton:hover, .md-button.myProfile_apiKey_refreshButton:focus {
- background-color: #D2D3D5; }
-
-.myProfile_form_removeButton {
- display: block;
- color: #FF3D4C;
- margin-bottom: 5px; }
-
-.myProfile_updateForm_error {
- position: absolute;
- top: 50px;
- left: 2px;
- color: red;
- font-size: 12px;
- line-height: 1; }
-
-.myProfile_apiKey_block {
- margin-top: 63px;
- margin-bottom: 65px; }
-
-.myProfile_apiKey {
- margin-top: 10px;
- margin-bottom: 10px; }
- .myProfile_apiKey svg {
- fill: #6E6E6E; }
-
-.md-button.myProfile_addKitButton:focus {
- background-color: #0019FF; }
-
-.md-button.myProfile_addKitButton:hover {
- background-color: #0019FF; }
-
-.myProfile_hiddenhref {
- position: absolute;
- height: 100%;
- width: 100%;
- top: 0;
- left: 0; }
-
-/**
- * TOOL LIST
- *
- */
-a.kitList h4 {
- color: #1C1C1C; }
-
-a.kitList p {
- color: #6E6E6E; }
-
-.clearfix {
- clear: both; }
-
-/*
- FOOTER COMPONENT
-*/
-footer {
- z-index: 1;
- width: 100%;
- background-color: #1C1C1C; }
-
-.footer_icon {
- display: block;
- width: 32px;
- height: 32px; }
-
-.flag {
- padding-right: 15px; }
-
-.border-white.footer-block {
- border: 2px white solid; }
-
-.uptimerobot-logo {
- width: 150px;
- justify-self: right;
- vertical-align: center;
- padding-top: 10px; }
-
-.uptimerobot-sponsor {
- display: flex;
- flex-direction: column;
- justify-content: start;
- align-items: flex-start; }
-
-.uptimerobot-sponsor-text p {
- font-size: 15px !important;
- margin: 0px; }
-
-@media only screen and (max-width: 950px) {
- .color-white {
- text-align: center; }
- .flag {
- padding-right: 0px; }
- .footer-block {
- margin-bottom: 20px !important; }
- .sponsor {
- justify-content: center;
- align-items: center; } }
-
-@media (max-width: 599px) {
- .layout-align-xs-start-stretch, .layout-align-xs-center-stretch, .layout-align-xs-end-stretch, .layout-align-xs-space-between-stretch, .layout-align-xs-space-around-stretch {
- align-items: center; } }
-
-/*
- RECOVERY PASSWORD STATE
-*/
-.recovery_header {
- width: 100%;
- height: 128px;
- background-color: #1C1C1C; }
-
-.recovery_headerIcon {
- display: block;
- width: 50px;
- height: 50px; }
-
-/*
- TIMELINE COMPONENT
-
- used on add and setup kit
-*/
-.timeline {
- height: 120px;
- background-color: #2E2E2E;
- position: fixed;
- width: 100%;
- z-index: 10;
- top: 64px;
- box-shadow: 0 4px 5px -2px #2E2E2E;
- padding: 30px; }
-
-@media (max-width: 960px) {
- .timeline-btn-extra {
- display: None; } }
-
-@media (min-width: 0px) and (max-width: 959px) and (orientation: portrait) {
- .timeline {
- top: 56px; }
- .profile_header {
- margin-top: 53px; } }
-
-@media (min-width: 0px) and (max-width: 959px) and (orientation: landscape) {
- .timeline {
- top: 53px; }
- .profile_header {
- margin-top: 53px; } }
-
-.timeline_container {
- width: 100%; }
-
-.timeline_line {
- width: 500px;
- border: 1px solid #FFC100;
- position: relative;
- top: 14px; }
-
-.timeline_line_small {
- width: 100px !important;
- margin: 0px 20px;
- top: 0px !important; }
-
-.timeline-title {
- text-transform: uppercase;
- color: #FFC100;
- font-size: 24px;
- font-weight: 600; }
- .timeline-title.vertical {
- margin-top: 10px; }
-
-.timeline_stepName {
- text-transform: uppercase;
- color: #FFC100;
- font-size: 18px; }
- .timeline_stepName.vertical {
- margin-top: 10px; }
-
-.timeline_content {
- min-height: 400px;
- width: 100%;
- margin-top: 220px; }
-
-.md-button.timeline_buttonBack {
- margin-left: auto;
- margin-right: 20px; }
-
-section.relaxed-layout {
- padding: 10px 25px; }
-
-/*
- ADD KIT STATE
-*/
-.kit_dataChange .form_block {
- padding: 40px 30px;
- background-color: #FFF; }
- .kit_dataChange .form_block.isEven {
- background-color: #F9F9FB; }
-
-.kit_dataChange .form_blockInput {
- height: inherit; }
-
-.kit_dataChange .form_blockInput_container {
- height: inherit; }
-
-.kit_dataChange .form_blockInput_button {
- height: inherit; }
-
-.kit_dataChange .form_blockInput_map {
- height: 250px;
- min-width: 250px; }
-
-.kit_dataChange .form_blockInput_select label {
- color: #0019FF;
- margin-right: 10px; }
-
-.kit_dataChange .form_blockInput_chips {
- margin-top: 35px; }
-
-.kit_dataChange .form_blockContent {
- padding-top: 5px;
- float: left; }
- .kit_dataChange .form_blockContent img, .kit_dataChange .form_blockContent div {
- float: left; }
- .kit_dataChange .form_blockContent img {
- border-radius: 50%;
- width: 120px;
- height: 120px;
- margin: 5px 20px; }
-
-.kit_dataChange .form_blockContent_image {
- position: relative;
- top: 3px; }
-
-.kit_dataChange .form_blockContent_text {
- margin-left: 10px;
- width: 240px; }
- .kit_dataChange .form_blockContent_text.long {
- width: auto !important; }
- .kit_dataChange .form_blockContent_text h2 {
- margin-top: 0;
- margin-bottom: 8px; }
- .kit_dataChange .form_blockContent_text p {
- font-size: 14px;
- color: #A5A5A5;
- line-height: 1.5; }
-
-.kit_dataChange .form_errors {
- position: absolute;
- bottom: 15%;
- left: 0;
- color: red;
- font-size: 0.7em;
- line-height: 1; }
-
-.emoji {
- font-weight: 400; }
-
-/*
- STATIC PAGES
-*/
-.static_page .timeline {
- align-items: flex-start stretch; }
- .static_page .timeline .content {
- padding: 0 0px; }
-
-.static_page .content {
- max-width: 50em;
- margin: 0 auto;
- text-align: justify;
- padding: 40px 10px; }
-
-.static_page .full-width-img {
- width: 100%;
- height: auto;
- max-width: 2000px;
- position: relative;
- margin: 0 0 -5px 0; }
-
-.static_page .embed-container {
- position: relative;
- padding-bottom: 56.25%;
- height: 0;
- overflow: hidden;
- max-width: 100%; }
- .static_page .embed-container iframe, .static_page .embed-container object, .static_page .embed-container embed {
- position: absolute;
- top: 0;
- left: 0;
- width: 100%;
- height: 100%; }
-
-.static_page .center {
- text-align: center;
- margin-top: 30px; }
-
-.static_page .row {
- min-height: 200px; }
- .static_page .row:nth-child(odd) {
- background-color: #F9F9FB; }
-
-.static_page h1 {
- color: white; }
-
-.static_page h4 {
- font-weight: bold;
- margin-bottom: 0.2em; }
-
-.static_page .md-button.md-primary.md-raised, .static_page .md-button.md-primary.md-fab {
- padding: 6px 20px;
- height: 44px;
- border-radius: 22px;
- font-size: 18px;
- background-color: #0019FF;
- color: white; }
- .static_page .md-button.md-primary.md-raised:hover, .static_page .md-button.md-primary.md-raised:focus, .static_page .md-button.md-primary.md-fab:hover, .static_page .md-button.md-primary.md-fab:focus {
- background-color: #3C98AF; }
- .static_page .md-button.md-primary.md-raised:not([disabled]), .static_page .md-button.md-primary.md-fab:not([disabled]) {
- box-shadow: none; }
-
-.static_page .md-button.md-primary {
- color: #0019FF; }
-
-.not-found-404 {
- height: 36vh; }
-
-@media (min-width: 960px) {
- .not-found-404 {
- height: 67.3vh; } }
-
-.sensor-image-margin {
- margin-top: 40px; }
-
-h3.no-margin {
- margin: 0; }
-
-p.no-margin {
- margin: 0; }
-
-.team-cells-margin {
- margin-bottom: 25px; }
-
-.subtitle-separation {
- margin-top: 60px; }
-
-.doorbell-button {
- background-color: #FFC100 !important; }
-
-.styleguide .info-text {
- color: rgba(0, 0, 0, 0.3); }
-
-.styleguide .info-text-dark {
- color: rgba(255, 255, 255, 0.3) !important; }
-
-.styleguide .dark-theme-divider {
- border-top-color: rgba(255, 255, 255, 0.12); }
-
-.styleguide small {
- margin-top: 20px; }
-
-.styleguide .dark-text-section {
- background-color: #1C1C1C; }
- .styleguide .dark-text-section h1 {
- font-size: 1.8em;
- color: #fff; }
- .styleguide .dark-text-section h2 {
- color: #fff; }
- .styleguide .dark-text-section h4 {
- color: #D2D3D5; }
- .styleguide .dark-text-section h6 {
- color: #87CCDD; }
- .styleguide .dark-text-section p {
- color: #C8E6ED;
- font-weight: 300; }
- .styleguide .dark-text-section small {
- color: #6E6E6E; }
-
-.styleguide .section-padding {
- padding: 60px; }
-
-.styleguide .colors-section p {
- padding-left: 20px; }
-
-.styleguide .colors-section .secondary-color {
- background-color: #6E6E6E; }
- .styleguide .colors-section .secondary-color p {
- color: #fff; }
-
-.styleguide .colors-section .terciary_color {
- background-color: #0019FF; }
- .styleguide .colors-section .terciary_color p {
- color: #fff; }
-
-.styleguide .colors-section .secondary_color_light {
- background-color: #8DB2BA; }
- .styleguide .colors-section .secondary_color_light p {
- color: #fff; }
-
-.styleguide .colors-section .secondary-color-pastel {
- background-color: #C8E6ED; }
- .styleguide .colors-section .secondary-color-pastel p {
- color: #8DB2BA; }
-
-.styleguide .colors-section .white {
- background-color: #fff;
- border: 1px solid #C8E6ED; }
- .styleguide .colors-section .white p {
- color: #8DB2BA; }
-
-.styleguide .colors-section .sensors > div {
- width: 60px;
- height: 60px;
- border-radius: 30px;
- float: left;
- margin: 10px; }
-
-.styleguide .colors-section .sensors div:nth-child(1) {
- background: #ffc107; }
-
-.styleguide .colors-section .sensors div:nth-child(2) {
- background: #4fc3f7; }
-
-.styleguide .colors-section .sensors div:nth-child(3) {
- background: #ffee58; }
-
-.styleguide .colors-section .sensors div:nth-child(4) {
- background: #f06292; }
-
-.styleguide .colors-section .sensors div:nth-child(5) {
- background: #4caf50; }
-
-.styleguide .colors-section .sensors div:nth-child(6) {
- background: #8bc34a; }
-
-.styleguide .colors-section .sensors div:nth-child(7) {
- background: #9575cd; }
-
-.styleguide .colors-section .sensors div:nth-child(8) {
- background: #fff9c4; }
-
-.styleguide .colors-section .sensors div:nth-child(9) {
- background: #ffee58; }
-
-.styleguide .colors-section .sensors div:nth-child(10) {
- background: #ff5722; }
-
-/* Custom Animations */
-.loading-dots span:after {
- content: '.';
- -webkit-animation: dots 2s steps(5, end) infinite;
- animation: dots 2s steps(5, end) infinite; }
-
-@-webkit-keyframes dots {
- 0%, 20% {
- color: rgba(0, 0, 0, 0);
- text-shadow: 0.25em 0 0 rgba(0, 0, 0, 0), 0.5em 0 0 rgba(0, 0, 0, 0); }
- 40% {
- color: white;
- text-shadow: 0.25em 0 0 rgba(0, 0, 0, 0), 0.5em 0 0 rgba(0, 0, 0, 0); }
- 60% {
- text-shadow: 0.25em 0 0 #FFC100, 0.5em 0 0 rgba(0, 0, 0, 0); }
- 80%, 100% {
- text-shadow: 0.25em 0 0 #FFC100, 0.5em 0 0 #0019FF; } }
-
-@keyframes dots {
- 0%, 20% {
- color: rgba(0, 0, 0, 0);
- text-shadow: 0.25em 0 0 rgba(0, 0, 0, 0), 0.5em 0 0 rgba(0, 0, 0, 0); }
- 40% {
- color: white;
- text-shadow: 0.25em 0 0 rgba(0, 0, 0, 0), 0.5em 0 0 rgba(0, 0, 0, 0); }
- 60% {
- text-shadow: 0.25em 0 0 #FFC100, 0.5em 0 0 rgba(0, 0, 0, 0); }
- 80%, 100% {
- text-shadow: 0.25em 0 0 #FFC100, 0.5em 0 0 #0019FF; } }
-
-/* Do not remove this comments bellow. It's the markers used by gulp-inject to inject
- all your sass files automatically */
-
-/*# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImluZGV4LnNjc3MiLCJoZWxwZXJzLnNjc3MiLCJfdmFyaWFibGVzLnNjc3MiLCJjb21wb25lbnRzL2xhbmRpbmcvbGFuZGluZy5zY3NzIiwiY29tcG9uZW50cy9sYXlvdXQvbGF5b3V0LnNjc3MiLCJjb21wb25lbnRzL3NlYXJjaC9zZWFyY2guc2NzcyIsImNvbXBvbmVudHMva2l0L2tpdC5zY3NzIiwiY29tcG9uZW50cy9raXQvc2hvd0tpdC9zaG93S2l0LnNjc3MiLCJjb21wb25lbnRzL3N0b3JlL3N0b3JlTW9kYWwuc2NzcyIsImNvbXBvbmVudHMvdXBsb2FkL2NzdlVwbG9hZC5zY3NzIiwiY29yZS9hbmltYXRpb24vYmFja2Ryb3AvbG9hZGluZ0JhY2tkcm9wLnNjc3MiLCJjb3JlL2FuaW1hdGlvbi9hbmltYXRpb24uZGlyZWN0aXZlLnNjc3MiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsbUpBQVk7QUFDWixpRkFBWTtBQ0FaO0VBQ0UsMEJDU2lCLEVEUmxCOztBQUNEO0VBQ0Usd0JBQXVCLEVBQ3hCOztBQUNEO0VBQ0Usd0JBQXVCLEVBQ3hCOztBQUNEO0VBQ0UsMEJBQStCLEVBQ2hDOztBQUNEO0VBQ0UsZUNIaUIsRURJbEI7O0FBQ0Q7RUFDRSxlQ2RZLEVEZWI7O0FBQ0Q7RUFDRSxlQ2pCWSxFRGtCYjs7QUFDRDtFQUNFLGVDaEJhLEVEb0JkO0VBTEQ7SUFHSSxjQ2xCVyxFRG1CWjs7QUFFSDtFQUNFLGVDdkJXLEVEMkJaO0VBTEQ7SUFHSSx5QkFBcUIsRUFDdEI7O0FBRUg7RUFDRSxZQUFXLEVBQ1o7O0FBQ0Q7RUFDRSxhQUFZLEVBQ2I7O0FBQ0Q7RUFDRSxjQUFZLEVBQ2I7O0FBQ0Q7RUFDRSxZQUFXLEVBQ1o7O0FBQ0Q7RUFDRSxpQkFBZ0IsRUFDakI7O0FBQ0Q7RUFDRSxnQkFBZSxFQUNoQjs7QUFHRDtFQUNFLGdDQUE4QixFQUMvQjs7QUFDRDtFQUNFLHFCQUFtQixFQUNwQjs7QUFFRDtFQUNFLG1CQUFrQixFQUNuQjs7QUFDRDtFQUNFLGlCQUFnQixFQUNqQjs7QUFLRDtFQUNFLHVCQUFzQixFQUN2Qjs7QUFDRDtFQUNFLHdCQUF1QixFQUN4Qjs7QUFDRDtFQUNFLDBCQ3ZFYSxFRHdFZDs7QUFDRDtFQUNFLDBCQzlFWSxFRCtFYjs7QUFDRDtFQUNFLHFDQUFpQyxFQUNsQzs7QUFDRDtFQUNFLHlDQ2pGVyxFRGtGWjs7QUFDRDtFQUNFLDBCQzFGYyxFRDJGZjs7QUFDRDtFQUNFLHFDQUFrQyxFQUNuQzs7QUFDRDtFQUNFLHFDQUEyQyxFQUM1Qzs7QUFFRDtFQUNFLHVCQUFzQixFQUN2Qjs7QUFHRDtFQUNFLDBCQ3RHWTtFRHVHWixlQ3ZHWTtFRHdHWixtQkFBaUIsRUFJbEI7RUFQRDtJQUtJLGNDMUdVLEVEMkdYOztBQUVIO0VBQ0UsK0JBQTRCO0VBQzVCLGFBQVksRUFJYjtFQU5EO0lBSUksWUFDRixFQUFDOztBQUVIO0VBQ0Usd0JBQXNCO0VBQ3RCLGFBQVk7RUFDWixtQkFBaUIsRUFDbEI7O0FBQ0Q7RUFDRSw2QkFBNEI7RUFDNUIsZUNuSGlCLEVEb0hsQjs7QUFDRDtFQUNFLHdCQUFzQjtFQUN0QixhQUFZO0VBQ1osbUJBQWlCLEVBQ2xCOztBQUNEO0VBQ0UsK0JBQTRCLEVBQzdCOztBQUNEO0VBQ0UsMEJDekljO0VEMElkLGVDMUljO0VEMklkLG1CQUFpQixFQUNsQjs7QUFDRDtFQUNFLCtCQUE4QjtFQUM5QixlQ3BJaUIsRURxSWxCOztBQUNEO0VBQ0UsMEJDbEpjO0VEbUpkLG9CQ25KYztFRG9KZCxlQ3pJaUI7RUQwSWpCLG1CQUFpQixFQUNsQjs7QUFFRDtFQUNFLCtCQUFpQztFQUNqQyxlQzFKYyxFRDJKZjs7QUFDRDtFQUNFLDBCQ2xKaUI7RURtSmpCLGVDbkppQjtFRG9KakIsbUJBQWlCLEVBQ2xCOztBQUNEO0VBQ0UscUNBQXVDO0VBQ3ZDLGFBQVksRUFDYjs7QUFDRDtFQUNFLDBCQ25LWTtFRG9LWixvQkNwS1k7RURxS1osYUFBWTtFQUNaLG1CQUFpQixFQUNsQjs7QUFFRDtFQUNFLGVDMUtZO0VEMktaLCtCQUFvQyxFQUNyQzs7QUFDRDtFQUNFLHdCQUFzQjtFQUN0QixpQkFBZ0I7RUFDaEIsYUFBWTtFQUNaLG1CQUFpQixFQUNsQjs7QUFFRDtFQUNFLGVDckxZO0VEc0xaLDZCQUE0QjtFQUM1QiwwQkN2TFksRUR3TGI7O0FBR0Q7RUFDRSxvQkFBbUI7RUFDbkIsZ0NBQStCO0VBQy9CLGdCQUFlO0VBRWYsa0JBQWlCO0VBQ2pCLGlCQ2hKcUI7RURpSnJCLHlCQUF3QjtFQUN4QixrQkFBZ0I7RUFDaEIscUhBQW9ILEVBQ3JIOztBQUVEO0VBQ0UsVUFBUztFQUNULG9CQUFtQjtFQUNuQixrQkFBZ0I7RUFDaEIsZ0JBQWMsRUFDZjs7QUFFRDtFQUNFLDBCQy9NWTtFRGdOWixhQUFXO0VBQ1gsMEJDak5ZLEVEa05iOztBQUVEO0VBQ0UsZUNyTlk7RURzTlosNkJBQTRCO0VBQzVCLDBCQ3ZOWSxFRHdOYjs7QUFHRDtFQUNFLGVBQWM7RUFDZCxZQUFXO0VBQ1gsbUJBQWtCO0VBQ2xCLFlBQVc7RUFDWCxtQkFBaUIsRUFDbEI7O0FBRUQ7RUFDRSw0QkFBMkI7RUFDM0IsNkJBQTRCLEVBRTdCOztBQUdEO0VBQ0UsVUFBUyxFQUNWOztBQUNEO0VBQ0UsYUFBWSxFQUNiOztBQUNEO0VBQ0UsaUJBQWdCLEVBQ2pCOztBQUNEO0VBQ0Usa0JBQWlCLEVBQ2xCOztBQUNEO0VBQVEsaUJBQWdCLEVBQUk7O0FBQzVCO0VBQVEsaUJBQWdCLEVBQUk7O0FBQzVCO0VBQVEsaUJBQWdCLEVBQUk7O0FBQzVCO0VBQVEsaUJBQWdCLEVBQUk7O0FBQzVCO0VBQVEsbUJBQWtCLEVBQUk7O0FBQzlCO0VBQVEsbUJBQWtCLEVBQUk7O0FBQzlCO0VBQVEsbUJBQWtCLEVBQUk7O0FBQzlCO0VBQVEsbUJBQWtCLEVBQUk7O0FBQzlCO0VBQVEsb0JBQW1CLEVBQUk7O0FBQy9CO0VBQVEsb0JBQW1CLEVBQUk7O0FBQy9CO0VBQVEsb0JBQW1CLEVBQUk7O0FBQy9CO0VBQVEsb0JBQW1CLEVBQUk7O0FBQy9CO0VBQVMsa0JBQWlCLEVBQUk7O0FBQzlCO0VBQVMsa0JBQWlCLEVBQUk7O0FBQzlCO0VBQVMsa0JBQWlCLEVBQUk7O0FBQzlCO0VBQVEsa0JBQWlCLEVBQUk7O0FBQzdCO0VBQ0UsNEJBQTJCO0VBQzNCLCtCQUE4QixFQUMvQjs7QUFDRDtFQUNFLGdCQUFlLEVBQ2hCOztBQUNEO0VBQU0sV0FBVSxFQUFJOztBQUNwQjtFQUFPLGNBQWEsRUFBSTs7QUFDeEI7RUFBTyxjQUFhLEVBQUk7O0FBQ3hCO0VBQU8sY0FBYSxFQUFJOztBQUN4QjtFQUFPLGNBQWEsRUFBSTs7QUFDeEI7RUFBTyxjQUFhLEVBQUk7O0FBQ3hCO0VBQU8sY0FBYSxFQUFJOztBQUN4QjtFQUFRLG1CQUFrQixFQUFJOztBQUM5QjtFQUFRLGtCQUFnQixFQUFJOztBQUM1QjtFQUNFLGtCQUFnQjtFQUNoQixxQkFBbUIsRUFDcEI7O0FBQ0Q7RUFDRSxtQkFBaUI7RUFDakIsb0JBQWtCLEVBQ25COztBQUNEO0VBQ0UsbUJBQWlCO0VBQ2pCLG9CQUFrQixFQUNuQjs7QUFLRDtFQUNFLG1CQUFrQixFQUNuQjs7QUFFRDtFQUVJLDJCQUF5QjtFQUN6Qiw2QkFBMkIsRUFDNUI7O0FBS0g7RUFDRSxtQkFBa0IsRUFDbkI7O0FBRUQ7RUFDRSxtQkFBa0IsRUFDbkI7O0FBR0Q7RUFDRSxZQUFXO0VBQ1gsYUFBWSxFQUNiOztBRWhVRDtFQUNFLGtEQUFpRDtFQUNqRCwwQkFBeUI7RUFDekIsc0JBQXFCLEVBQ3RCOztBQUVEO0VBRUkseUJBQXdCO0VBQ3hCLHFCQUFvQjtFQUNwQixpQkFBZ0I7RUFDaEIsZURGZSxFQ0doQjs7QUFOSDtFQVFJLGdCQUFlO0VBQ2Ysa0JBQWlCO0VBQ2pCLGdCQUFjO0VBQ2Qsa0JBQWlCLEVBQ2xCOztBQVpIO0VBY0ksZ0JBQWU7RUFDZixrQkFBaUI7RUFDakIsZ0JBQWMsRUFDZjs7QUFqQkg7RUFtQkksZ0JBQWU7RUFDZixrQkFBaUI7RUFDakIsaUJBQWU7RUFDZixvQkFBbUIsRUFDcEI7O0FBdkJIO0VBeUJJLGVEdEJlO0VDdUJmLGdDQUErQjtFQUMvQixnQkFBZTtFQUNmLGtCQUFpQixFQUNsQjs7QUE3Qkg7RUErQkksaUNBQWdDLEVBQ2pDOztBQWhDSDtFQWtDSSxhQUFZO0VBQ1osbUJBQWtCO0VBQ2xCLFdBQVU7RUFDVixVQUFTO0VBQ1QsVUFBUSxFQUNUOztBQXZDSDtFQXlDSSxtQkFBa0I7RUFDbEIsV0FBVTtFQUNWLFdBQVU7RUFDVixVQUFRLEVBRVQ7O0FBOUNIO0VBaURJLGlCQUFnQjtFQUNoQixnRUFBK0QsRUFDaEU7O0FBbkRIO0VBcURJLGlCQUFnQjtFQUNoQixnRUFBK0QsRUFDaEU7O0FBdkRIO0VBeURJLHdFQUF1RSxFQUN4RTs7QUExREg7RUE0REkscUVBQW9FLEVBQ3JFOztBQTdESDtFQStESSxtRUFBa0UsRUFDbkU7O0FBaEVIO0VBa0VJLDZCQUEyQjtFQUMzQixxRUFBb0UsRUFDckU7O0FBcEVIO0VBc0VJLDZCQUEyQjtFQUMzQiw4REFBNkQsRUFDOUQ7O0FBeEVIO0VBMkVJLG1CQUFrQjtFQUNsQixpQ0R6RWU7RUMwRWYsZ0NEMUVlLEVDMkVoQjs7QUE5RUg7RUFnRkksa0JBQWdCO0VBQ2hCLHNCQUFvQixFQUNyQjs7QUFsRkg7RUFxRkksK0JEbEZlLEVDbUZoQjs7QUF0Rkg7RUF3RkksOEJEckZlLEVDc0ZoQjs7QUF6Rkg7RUE0RkksaUJBQWdCLEVBQ2pCOztBQTdGSDtFQWtHSSw0RkFBMkY7RUFDM0YsbUJBQWtCO0VBQ2xCLGFBQVk7RUFDWixZQUFXO0VBQ1gsaUJBQWdCLEVBQ2pCOztBQXZHSDtFQXlHSSxtQkFBa0I7RUFDbEIsV0FBVTtFQUNWLFNBQVE7RUFDUixhQUFZO0VBQ1osa0JBQWlCO0VBQ2pCLG1CQUFrQixFQUtuQjtFQW5ISDtJQWlITSxxQkFBbUIsRUFDcEI7O0FBR0g7RUFySEY7SUF1SE0sZ0JBQWM7SUFDZCxrQkFBZ0IsRUFDakI7RUF6SEw7SUEySE0sZ0JBQWM7SUFDZCxrQkFBZ0IsRUFDakI7RUE3SEw7SUErSE0sZ0JBQWU7SUFDZixrQkFBaUIsRUFDbEI7RUFqSUw7SUFtSU0sZ0JBQWM7SUFDZCxrQkFBZ0IsRUFDakI7RUFySUw7SUF1SU0sU0FBTyxFQUNSO0VBeElMO0lBMElNLGtCQUFpQjtJQUNqQixtQkFBa0I7SUFDbEIsaUJBQWdCO0lBQ2hCLG1CQUFpQjtJQUNqQixhQUFZLEVBSWI7SUFsSkw7TUFnSlEscUJBQW1CLEVBQ3BCLEVBQUE7O0FBS0w7RUF0SkY7SUF3Sk0sZ0JBQWMsRUFDZjtFQXpKTDtJQTJKTSxnQkFBYyxFQUNmO0VBNUpMO0lBOEpNLG1CQUFrQixFQUNuQixFQUFBOztBQUdIO0VBbEtGO0lBb0tNLGdCQUFjO0lBQ2Qsa0JBQWdCO0lBQ2hCLHFCQUFtQixFQUNwQjtFQXZLTDtJQXlLTSxnQkFBYyxFQUNmO0VBMUtMO0lBNEtNLGdCQUFjLEVBQ2Y7RUE3S0w7SUErS00sOEJBQTZCLEVBQzlCO0VBaExMO0lBa0xNLDhCRC9LYTtJQ2dMYiwrQkRoTGE7SUNpTGIsZ0NEakxhO0lDa0xiLGlDRGxMYSxFQ21MZDtFQXRMTDtJQXdMTSxrQkFBaUI7SUFDakIsK0JEdExhO0lDdUxiLGdDRHZMYTtJQ3dMYixpQ0R4TGEsRUN5TGQ7RUE1TEw7SUE4TE0sNkJBQTJCLEVBQzVCO0VBL0xMO0lBaU1NLGtCQUFpQixFQUNsQjtFQWxNTDtJQW9NTSxtQkFBa0IsRUFDbkI7RUFyTUw7SUF1TU0sNEJBQTJCO0lBQzNCLCtCQUE4QixFQUMvQjtFQXpNTDtJQTJNTSxnRUFBK0QsRUFDaEU7RUE1TUw7SUE4TU0sZ0VBQStELEVBQ2hFLEVBQUE7O0FDdk5MOztHQUVHO0FBRUg7RUFHTSxhRndCa0IsRUV2Qm5COztBQUpMO0VBUUksMEJGWlk7RUVhWixlRmJZO0VFY1osb0JBQW1CO0VBR25CLG9CQUFtQjtFQUNuQix1QkFBc0I7RUFDdEIsZ0hBQStHLEVBS2hIO0VBcEJIO0lBaUJNLHFDQUFtQztJQUNuQywwQkFBNEIsRUFDN0I7O0FBbkJMO0VBdUJJLG1CQUFrQixFQU1uQjtFQTdCSDtJQTBCTSxhQUFZO0lBQ1osWUFBVyxFQUNaOztBQTVCTDtFQWdDSSxrQkFBaUI7RUFDakIsd0JBQXVCO0VBQ3ZCLHVCQUFzQixFQUl2QjtFQXRDSDtJQW9DTSxjRnhDVSxFRXlDWDs7QUFyQ0w7RUEwQ00saUJBQWdCO0VBQ2hCLDBCRi9DVTtFRWdEVixlRmhEVTtFRWlEVixvQkFBbUI7RUFDbkIsWUFBVyxFQUNaOztBQS9DTDtFQWlETSwwQkZyRFU7RUVzRFYsZUYzQ2EsRUU0Q2Q7O0FBS0w7RUFDRSxZQUFXO0VBQ1gsYUFBWTtFQUNaLG9CQUFtQixFQUNwQjs7QUFHRDtFQUNFLDRCQUEyQixFQUM1Qjs7QUNoRUQ7RUFDSSx3REFBdUQ7RUFDdkQsNkJBQTRCO0VBQzVCLCtCQUE4QjtFQUM5QixzQkFBcUI7RUFDckIsa0JBQWlCLEVBQ3BCOztBQUVEO0VBQ0UsY0FBYTtFQUNiLG9CQUFtQixFQVlwQjtFQWREO0lBSUksZUFBYztJQUNkLG1CQUFrQjtJQUNsQixhQUFZLEVBQ2I7RUFQSDtJQVNJLG1CQUFrQixFQUNuQjtFQVZIO0lBWUksZUh0QlUsRUd1Qlg7O0FDMUJIOztHQUVHO0FBRUg7RUFDRSxrQkFBaUIsRUFDbEI7O0FBRUQ7RUFDRSxXSnNDaUI7RUlyQ2pCLG1CQUFrQixFQUNuQjs7QUFFRDtFQUNFLFlBQVU7RUFDVixrQkFBaUIsRUEwQ2xCO0VBNUNEO0lBS0ksWUFBVTtJQUNWLGFBQVc7SUFDWCxtQkFBaUI7SUFDakIsd0JBQXVCO0lBQ3ZCLFdBQVU7SUFDVixlSlplO0lJYWYsbUJBQWlCO0lBQ2pCLGdCQUFjO0lBQ2QscUJBQW9CLEVBV3JCO0lBeEJIO01BZ0JNLGVKbEJhO01JbUJiLGlCQUFnQjtNQUNoQixVQUFRLEVBQ1Q7SUFuQkw7TUFxQk0sZUpuQlE7TUlvQlIsY0FBYSxFQUNkO0VBdkJMO0lBMkJJLHdCQUF1QjtJQUN2QixtQkFBaUI7SUFDakIsZUozQlU7SUk2QlYsZ0JBQWM7SUFFZCxZQUFXO0lBQ1gsbUJBQWlCO0lBQ2pCLGFBQVc7SUFDWCxZQUFVLEVBT1g7SUEzQ0g7TUFzQ00sZUpuQ2UsRUlvQ2hCO0lBdkNMO01BeUNNLGlCQUFlLEVBQ2hCOztBQUlMO0VBRUksU0FBUTtFQUNSLFlBQVc7RUFDWCxxQ0FBaUM7RUFDakMsbUJBQWtCO0VBQ2xCLCtDQUEyQztFQUMzQyxXQUFVLEVBQ1g7O0FBRUg7RUFDRSxhQUFZO0VBQ1osY0FBYSxFQUNkOztBQUNEO0VBQ0UsYUFBWTtFQUNaLDhCQUE2QjtFQUM3QixtQkFBa0I7RUFDbEIsYUFBWTtFQUNaLFlBQVcsRUFDWjs7QUMvRUQ7Ozs7RUFJRTtBQUNGO0VBQ0Usc0JBQXFCLEVBQ3RCOztBQUVEOztFQUVFO0FBRUY7RUFDRSxxQkFBcUI7RUFDckIsa0JBQWlCO0VBQ2pCLGdCQUFnQjtFQUNoQixnQkFBZ0I7RUFDaEIsbUJBQWtCLEVBa0VuQjtFQXZFRDtJQVFJLFdBQVU7SUFDVixrQkFBaUIsRUFDbEI7RUFWSDtJQVlJLGFBQVksRUFDYjtFQWJIO0lBZ0JJLFdBQVU7SUFDVixnQkxkaUI7SUtlakIsZ0JBQWU7SUFDZiw0QkFBMkI7SUFDM0IsY0FBYSxFQUNkO0VBckJIO0lBdUJJLGdCQUFlLEVBQ2hCO0VBeEJIO0lBMEJJLGNMeEJVLEVLeUJYO0VBM0JIO0lBNkJJLGNMM0JVLEVLNEJYO0VBOUJIO0lBZ0NJLGNMOUJVLEVLK0JYO0VBakNIO0lBb0NNLGdCTGxDUTtJS21DUixvQkFBbUI7SUFDbkIsNEJBQTJCLEVBQzVCO0VBdkNMO0lBeUNNLGdCQUFlLEVBQ2hCO0VBMUNMO0lBNkNJLFdBQVU7SUFDVixvQkFBbUIsRUFDcEI7RUEvQ0g7SUFrREksY0FBb0I7SUFDcEIsa0JBQWlCLEVBQ2xCO0VBcERIO0lBc0RJLDBCTG5EaUIsRUtvRGxCO0VBdkRIO0lBeURJLGNMM0RlO0lLNERmLGVMNURlO0lLNkRmLGdCQUFlO0lBQ2Ysa0JBQWlCLEVBQ2xCO0VBN0RIO0lBK0RJLGNMNURpQjtJSzZEakIsZ0JBQWU7SUFDZixrQkFBaUIsRUFDbEI7RUFsRUg7SUFvRUksY0xsRVU7SUttRVYsa0JBQWlCLEVBQ2xCOztBQUdIO0VBQ0UsZ0JBQWU7RUFDZixTQUFRLEVBQ1Q7O0FBRUQ7RUFDRSxrQkFBaUI7RUFDakIsYUFBWTtFQUNaLFlBQVc7RUFDWCxtQkFBaUI7RUFDakIsYUFBVyxFQUNaOztBQUdEOztHQUVHO0FBQ0g7RUFDRSxZQUFXO0VBQ1gsV0wzRGU7RUs0RGYsMEJMNUZvQjtFSzZGcEIsbUJBQWtCO0VBdUZsQixtQkFBbUI7RUFzRG5CLG9CQUFvQixFQXFDckI7RUF0TEQ7SUFPSSxZQUFXO0lBQ1gsYUFBWSxFQUNiO0VBVEg7SUFZSSxtQkFBa0I7SUFFbEIsbUJBQWtCO0lBQ2xCLGVMdEdpQjtJS3VHakIsc0JBQXFCO0lBQ3JCLGdCQUFlO0lBQ2YsMEJBQXlCLEVBQzFCO0VBbkJIO0lBcUJJLGtCQUFnQjtJQUVoQixrQkFBaUI7SUFFakIsZUxoSGlCO0lLaUhqQixrQkFBaUI7SUFDakIsZ0JBQWUsRUFlaEI7SUExQ0g7TUE4Qk0sbUJBQWlCO01BQ2pCLFlBQVU7TUFDVixZQUFXO01BQ1gsYUFBWTtNQUNaLGtCQUFpQjtNQUNqQixnQkFBZSxFQUNoQjtJQXBDTDtNQXVDTSxlTDlIZTtNSytIZixzQkFBcUIsRUFDdEI7RUF6Q0w7SUE0Q0ksa0JBQWdCO0lBRWhCLGtCQUFpQjtJQUNqQixnQkFBZTtJQUNmLGtCQUFpQjtJQUNqQixlTHhJaUIsRUtzSmxCO0lBL0RIO01BcURNLFlBQVU7TUFDVixZQUFXO01BQ1gsYUFBWTtNQUNaLGtCQUFpQjtNQUNqQixnQkFBZSxFQUVoQjtNQTNETDtRQTBEWSxjTGpLSSxFS2lLYTtJQTFEN0I7TUE2RE0sdUJBQXNCLEVBQ3ZCO0VBOURMO0lBa0VJLGtCQUFnQjtJQUVoQixrQkFBaUI7SUFDakIsZUw1SmlCO0lLNkpqQixnQkFBZTtJQUNmLGtCQUFpQixFQVVsQjtJQWpGSDtNQTBFTSxtQkFBa0I7TUFDbEIsWUFBVztNQUNYLGtCQUFpQixFQUlsQjtNQWhGTDtRQThFUSxjTHJLYSxFS3NLZDtFQS9FUDtJQW9GSSxrQkFBaUIsRUFDbEI7RUFyRkg7SUF3RkksdUJBQXNCLEVBQ3ZCO0VBekZIO0lBNkZJLHVCQUFzQjtJQUN0QixtQkFBa0I7SUFDbEIsc0JBQXFCO0lBQ3JCLFlBQVc7SUFDWCxhQUFZLEVBQ2I7RUFsR0g7SUFxR0ksV0FBVTtJQUNWLFNBQVE7SUFDUixVQUFTLEVBQ1Y7RUF4R0g7SUEyR0ksbUJBQWtCO0lBQ2xCLHVCQUFzQjtJQUN0QixnQkFBZTtJQUNmLE9BQU07SUFDTixRQUFPO0lBQ1AsU0FBUTtJQUNSLFVBQVM7SUFDVCx1QkFBc0I7SUFFdEIsZ0JBQWUsRUFDaEI7RUFySEg7SUF3SEksbUJBQWtCO0lBQ2xCLFlBQVc7SUFDWCxhQUFZO0lBQ1osWUFBVztJQUNYLFVBQVM7SUFDVCxZQUFXO0lBQ1gsd0JBQXVCO0lBRXZCLGdCQUFlLEVBQ2hCO0VBaklIO0lBb0lJLDBCTDNPWSxFSzRPYjtFQXJJSDtJQXdJSSw0QkFBMkIsRUFDNUI7RUF6SUg7SUE0SUksb0NBQW1DO0lBRW5DLDRCQUEyQixFQUM1QjtFQS9JSDtJQW1KSSxvQkFBbUIsRUFDcEI7RUFwSkg7SUF1SkksbUJBQWtCLEVBQ25CO0VBeEpIO0lBMkpJLHVCQUFzQjtJQUN0QixlTG5QaUI7SUtvUGpCLGdCQUFlO0lBQ2Ysa0JBQWlCLEVBQ2xCO0VBL0pIO0lBa0tJLG1CQUFrQixFQWtCbkI7SUFwTEg7TUFzS00sWUFBVyxFQUlaO01BMUtMO1FBd0tRLGNML1BhLEVLZ1FkO0lBektQO01BNktNLFVBQVEsRUFDVDtJQTlLTDtNQWlMTSxpQ0w3UWEsRUs4UWQ7O0FBTUw7RUFDRSxZQUFXO0VBQ1gsYUFBWTtFQUNaLFdMblBpQjtFS29QakIsMEJMdFJxQjtFS3VSckIsbUJBQWtCLEVBa0duQjtFQXZHRDtJQVFJLFlBQVc7SUFLWCxpQkFBZ0IsRUFDakI7RUFkSDtJQWlCSSxRQUFPLEVBQ1I7RUFsQkg7SUFxQkksU0FBUSxFQUNUO0VBdEJIO0lBeUJJLFdBQVU7SUFDVixhQUFZO0lBQ1osZUFBYztJQUNkLG1CQUFrQjtJQUNsQixtQkFBa0IsRUFDbkI7RUFFRDtJQWhDRjtNQWtDTSxpQkFBZ0IsRUFDakIsRUFBQTtFQW5DTDtJQXVDSSxtQkFBa0I7SUFDbEIsZUx2VGlCO0lLd1RqQixzQkFBcUI7SUFDckIsZ0JBQWU7SUFDZixtQkFBa0I7SUFDbEIsY0FBWSxFQTBEYjtJQXRHSDtNQStDTSxZQUFXO01BQ1gscUJBQW9CO01BQ3BCLGlCQUFnQjtNQUNoQixvQkFBbUIsRUFDcEI7SUFuREw7TUFzRE0sWUFBVyxFQUNaO0lBdkRMO01BMERNLG1CQUFrQjtNQUNsQixZQUFXO01BQ1gsYUFBWTtNQUNaLGdCQUFlLEVBYWhCO01BMUVMO1FBK0RRLGdCQUFjO1FBQ2QsZ0JBQWUsRUFDaEI7TUFqRVA7UUFtRVEsa0JBQWdCO1FBQ2hCLFlBQVc7UUFDWCxZQUFXLEVBSVo7UUF6RVA7VUFzRXlCLGNMOVZWLEVLOFYwQjtRQXRFekM7VUF1RTJCLGNMaFdkLEVLZ1c0QjtRQXZFekM7VUF3RXNCLGNMbldGLEVLbVd1QjtJQXhFM0M7TUE0RU0saUJBQWdCO01BQ2hCLG1CQUFrQjtNQUNsQixvQkFBbUIsRUFDcEI7SUEvRUw7TUFpRk0sY0FBYSxFQUNkO0lBbEZMO01Bb0ZNLFlBQVc7TUFDWCxhQUFZO01BQ1osbUJBQWtCO01BQ2xCLGFBQVk7TUFDWixXQUFVLEVBSVg7TUE1Rkw7UUEwRlEsY0x6V2EsRUswV2Q7SUEzRlA7TUE4Rk0sYUFBWSxFQUNiO0lBL0ZMO01BaUdNLGVMclhhLEVLeVhkO01BckdMO1FBbUdRLGNMdlhXLEVLd1haOztBQUtQOzs7R0FHRztBQU1IO0VBQ0UsVUFBUyxFQUNUOztBQUNGO0VBQ0UsZUxyWW1CLEVLc1lwQjs7QUFDRDtFQUNFLGFBQVksRUFDYjs7QUFDRDtFQUNFLG1CQUFtQjtFQUNuQixXTDdXaUI7RUs4V2pCLHdCQUF1QjtFQUN2QixtQkFBa0IsRUFxWm5CO0VBelpEO0lBT0kscUJBQW9CO0lBQ3BCLHFCQUFvQjtJQUNwQixXQUFVLEVBQ1g7RUFWSDtJQWFJLFVBQVM7SUFDVCxpQkFBZ0IsRUFDakI7RUFmSDtJQWtCSSxrQkFBaUI7SUFDakIsd0JBQXVCO0lBQ3ZCLG1CQUFrQixFQThObkI7SUFsUEg7TUF1Qk0sMEJMdGFhO01LdWFiLGFBQVksRUFDYjtJQXpCTDtNQTRCTSxjQUFhO01BQ2IsYUFBWSxFQW9OYjtNQWpQTDtRQWdDUSxlTC9hVyxFS21oQlo7UUFwSVA7VUFvQ1ksWUFBVTtVQUNWLGtCQUFpQjtVQUNqQixpQkFBZSxFQUloQjtVQTFDWDtZQXdDYyxjTHZiSyxFS3diTjtRQXpDYjtVQTRDWSxVQUFTLEVBU1Y7VUFyRFg7WUFnRGtCLGNBQVksRUFFYjtRQWxEakI7VUF3RFUsb0NBQW9DO1VBQ3BDLG1CQUFtQjtVQUNuQiw0QkFBNEI7VUFDNUIseUJBQXlCLEVBaUMxQjtVQTVGVDtZQThEWSxVQUFTO1lBQ1QsbUJBQWlCO1lBQ2pCLGVBQWMsRUFDZjtVQWpFWDtZQW9FWSxpQkFBZ0IsRUFDakI7VUFyRVg7WUF3RVksa0JBQWlCO1lBQ2pCLHNCQUFxQjtZQUNyQixlTHBkUyxFS3FkVjtVQTNFWDtZQThFWSxtQkFBa0I7WUFDbEIsV0FBVTtZQUNWLGlCQUFnQjtZQUNoQixjQUFhO1lBQ2Isa0JBQWlCO1lBQ2pCLGFBQVk7WUFFWiwyQ0FBMEM7WUFDMUMsZUxyZU87WUtzZVAsbUJBQWtCO1lBQ2xCLG1CQUFrQjtZQUNsQixjQUFhO1lBQ2IsMENBQXlDLEVBQzFDO1FBM0ZYO1VBZ0dZLGVBQWM7VUFDZCxrQkFBaUIsRUFDbEI7UUFsR1g7VUFvR1ksZ0JBQWU7VUFDZixtQkFBa0I7VUFDbEIsVUFBUyxFQUNWO1FBdkdYO1VBeUdZLFlBQVc7VUFDWCxhQUFZO1VBQ1osaUJBQWdCLEVBQ2pCO1FBNUdYO1VBK0dVLGdCQUFlO1VBQ2YsY0FBYTtVQUNiLG1CQUFrQjtVQUNsQix1Q0FBc0M7VUFJdEMsK0JBQThCO1VBQzlCLDJDQUEwQztVQUMxQyxlTHZnQlM7VUt3Z0JULG1CQUFrQjtVQUNsQixtQkFBa0I7VUFDbEIsaUJBQWdCO1VBQ2hCLDBDQUF5QztVQUN6QyxpQkFBZ0IsRUFDakI7UUE5SFQ7VUFnSVUsZ0JBQWU7VUFDZixrQkFBZ0I7VUFDaEIsaUJBQWdCLEVBQ2pCO01BbklUO1FBdUlRLDRCQUE0QjtRQUM1QixtQkFBbUIsRUF3R3BCO1FBaFBQO1VBMklVLFlBQVc7VUFDWCxnQkFBZSxFQW1HaEI7VUEvT1Q7WUE4SVksa0JBQWlCLEVBQ2xCO1VBL0lYO1lBaUpZLGtCQUFpQjtZQUNqQixtQkFBa0I7WUFDbEIsZ0JBQWUsRUFDaEI7VUFwSlg7WUF1SmMsWUFBVztZQUNYLGFBQVk7WUFDWixrQkFBaUIsRUFDbEI7VUExSmI7WUE4Slksa0JBQWlCLEVBQ2xCO1VBRUQ7WUFqS1Y7Y0FtS2MseUNBQXdDO2NBQ3hDLHVCQUFzQixFQUN2QjtZQXJLYjtjQXVLYywrQkFBOEIsRUFDL0IsRUFBQTtVQUdIO1lBM0tWO2NBOEtjLHlCQUF3QjtjQUN4QixpQ0FBZ0M7Y0FDaEMsOEJBQTZCO2NBQzdCLHVCQUFzQjtjQUN0QiwwQ0FBeUM7Y0FDekMsK0JBQThCLEVBQy9CO1lBcExiO2NBdUxjLGlDQUFnQztjQUNoQyx1QkFBc0IsRUFDdkI7WUF6TGI7Y0EyTGMseUJBQXdCO2NBQ3hCLHVCQUFzQjtjQUN0QiwwQ0FBeUM7Y0FDekMsNEJBQTJCLEVBSzVCO2NBbk1iO2dCQWlNZ0IsdUJBQXNCLEVBQ3ZCLEVBQUE7VUFsTWY7WUF3TWMsd0JBQXVCO1lBQ3ZCLFlBQVc7WUFDWCxhQUFZO1lBQ1oscUJBQW9CO1lBQ3BCLDBCTHRsQk87WUt1bEJQLDBCTDFsQlMsRUsybEJWO1VBOU1iO1lBaU5jLDRCQUEyQjtZQUMzQiwrQkFBOEIsRUFVL0I7WUE1TmI7Y0FvTmdCLGNMOWxCSyxFSytsQk47WUFyTmY7Y0F1TmdCLDBCTGptQkssRUtxbUJOO2NBM05mO2dCQXlOa0IsY0x0bUJLLEVLdW1CTjtVQTFOakI7WUErTmdCLGNMem1CSyxFSzBtQk47VUFoT2Y7WUFrT2dCLDBCTDVtQkssRUtnbkJOO1lBdE9mO2NBb09rQixjTGpuQkssRUtrbkJOO1VBck9qQjtZQXlPYyxhQUFZO1lBQ1osWUFBVztZQUNYLG1CQUFrQjtZQUNsQixTQUFRLEVBQ1Q7RUE3T2I7SUFvUEksZUxub0JlLEVLb29CaEI7RUFyUEg7SUF3UE0sY0x2b0JhLEVLd29CZDtFQXpQTDtJQTRQTSxpQkFBZSxFQUNoQjtFQTdQTDtJQWdRSSxxQkFBb0IsRUFDckI7RUFqUUg7SUFvUU0sbUJBQWtCLEVBQ25CO0VBclFMO0lBd1FJLGNMbHBCaUIsRUttcEJsQjtFQXpRSDtJQTRRTSxlTHRwQmU7SUt1cEJmLGtCQUFpQixFQUNsQjtFQTlRTDtJQWtSTSxlTDVwQmU7SUs2cEJmLGtCQUFpQixFQUNsQjtFQXBSTDtJQXdSSSxpQkFBZ0IsRUFDakI7RUF6Ukg7SUEyUkksZ0JBQWU7SUFDZixvQkFBbUIsRUFDcEI7RUE3Ukg7SUErUkksZ0JBQWUsRUFDaEI7RUFoU0g7SUFrU0ksb0JBQW1CLEVBQ3BCO0VBblNIO0lBcVNJLGVMenJCUyxFSzByQlY7RUF0U0g7SUE0U0ksWUFBVztJQUNYLGFBQVksRUFDYjtFQTlTSDtJQWdUSSwwQkFBeUIsRUFDMUI7RUFqVEg7SUFtVEksWUFBVztJQUNYLGFBQVk7SUFDWixrQkFBaUI7SUFDakIsbUJBQWtCO0lBQ2xCLFVBQVMsRUFJVjtJQTNUSDtNQXlUTSxjTG5zQmUsRUtvc0JoQjtFQTFUTDtJQTZUSSxZQUFXO0lBQ1gsYUFBWTtJQUNaLGtCQUFpQjtJQUNqQixtQkFBa0I7SUFDbEIsVUFBUyxFQUNWO0VBbFVIO0lBb1VJLGdCQUFlLEVBQ2hCO0VBclVIO0lBdVVJLGdCQUFlO0lBQ2YsZ0JBQWUsRUFDaEI7RUF6VUg7SUEyVUksc0JBQXFCLEVBTXRCO0VBalZIO0lBbVZJLGVMbHVCZTtJS211QmYsMEJBQXlCO0lBQ3pCLHFCQUFtQixFQW1FcEI7SUF4Wkg7TUF3Vk0sV0FBVTtNQUNWLGVBQWMsRUFDZjtJQTFWTDtNQTRWTSxzQkFBcUIsRUFDdEI7SUE3Vkw7TUErVk0sZUw5dUJhO01LK3VCYixzQkFBcUIsRUFDdEI7SUFqV0w7TUFtV00sZUw3dUJlO01LOHVCZixrQkFBaUIsRUFDbEI7SUFyV0w7TUF1V00sc0JBQXFCO01BQ3JCLGVMdnZCYSxFS3d2QmQ7SUF6V0w7TUEyV00sb0JBQW1CLEVBQ3BCO0lBNVdMO01BOFdNLDBCQUF5QjtNQUN6QixhQUFZO01BQ1osd0JBQXVCLEVBQ3hCO0lBalhMO01BbVhNLFlBQVc7TUFDWCxZQUFXO01BQ1gsYUFBWTtNQUNaLGlCQUFnQjtNQUNoQixnQkFBZSxFQUNoQjtJQXhYTDtNQTBYTSxZQUFXO01BQ1gsZ0JBQWUsRUFDaEI7SUE1WEw7TUE4WE0sZ0JBQWUsRUFJaEI7TUFsWUw7UUFnWVEsbUJBQWtCLEVBQ25CO0lBallQO01Bb1lNLGdCQUFlO01BQ2YsZ0JBQWU7TUFDZixvQkFBbUI7TUFDbkIsa0JBQWlCO01BQ2pCLGFBQVksRUFDYjtJQXpZTDtNQTJZTSwwQkxyeUJVLEVLc3lCWDtJQTVZTDtNQThZTSwwQkxseUJPLEVLbXlCUjtJQS9ZTDtNQWlaTSxhQUFZO01BQ1osaUJBQWdCLEVBQ2pCO0lBblpMO01BcVpNLFlBQVc7TUFDWCxhQUFZLEVBQ2I7O0FBSUw7RUFDRSxnQkFBZTtFQUNmLG1CQUFrQjtFQUNsQixpQkFBZ0IsRUFDakI7O0FBRUQsSUFBSTtBQUVKOzhDQUM4QztBQUc5QztFQUNFLDRCQUEwQixFQUMzQjs7QUFDRDtFQUNFLCtCQUFzQjtVQUF0Qix1QkFBc0IsRUFFdkI7O0FBQ0Q7RUFDRSw0QkFBbUI7VUFBbkIsb0JBQW1CLEVBRXBCOztBQUVEOzs7Ozs7OztJQVFJO0FBRUoseUNBQXlDO0FBRXpDO0VBQ0UsNEJBQTBCLEVBQzNCOztBQUNEO0VBQ0UsK0JBQXNCO1VBQXRCLHVCQUFzQixFQUN2Qjs7QUFDRDtFQUNFLDRCQUFtQjtVQUFuQixvQkFBbUIsRUFDcEI7O0FBRUQ7Ozs7Ozs7O0lBUUk7QUFFSjtFQUNFLDRCQUEwQixFQUMzQjs7QUFDRDtFQUNFLCtCQUFzQjtVQUF0Qix1QkFBc0IsRUFDdkI7O0FBQ0Q7RUFDRSw0QkFBbUI7VUFBbkIsb0JBQW1CLEVBQ3BCOztBQUVEOzs7Ozs7OztJQVFJO0FDLzNCSjs7RUFFRTtBQUVGO0VBQ0Usa0JBQWdCLEVBT2pCO0VBUkQ7SUFHSSwyREFBMEQ7SUFDMUQsNkJBQTRCO0lBQzVCLGdCQUFlO0lBQ2YsbUJBQWtCLEVBQ25COztBQ1hIO0VBQ0UsY0FBYTtFQUNiLG9CQUFtQixFQU9wQjtFQVREO0lBSUksbUJBQWtCLEVBQ25CO0VBTEg7SUFPSSxtQkFDRixFQUFDOztBQUdIO0VBQ0UsK0NBQTJDLEVBTzVDO0VBUkQ7SUFHSSxpQ1BBa0IsRU9DbkI7RUFKSDtJQU1JLG9CQUFtQixFQUNwQjs7QUFPSDtFQUNFLDBCUFhZLEVPWWI7O0FBRUQ7RUFDRSxnQkFBZSxFQUNoQjs7QUFFRCx1QkFBdUI7QUFFdkI7RUFHTSxtQkFBa0I7RUFDbEIsa0JBQWlCLEVBQ2xCOztBQUxMO0VBUU0sa0JBQWlCO0VBQ2pCLGlCQUFnQixFQUNqQjs7QUFFRDtFQVpKO0lBY1ksYUFBWSxFQUNmLEVBQUE7O0FBR0w7RUFsQko7SUFvQlksYUFBWSxFQUNmLEVBQUE7O0FBR0w7RUF4Qko7SUEwQlksY0FBYSxFQUNoQixFQUFBOztBQzlEVDs7R0FFRztBQUlIO0VBQ0Usd0NBQXVDLEVBQ3hDOztBQUlEO0VBQ0UsY0FBYTtFQUNiLG9CQUFtQjtFQUNuQix3QkFBdUI7RUFDdkIsYUFBWTtFQUNaLGNBQWEsRUFLZDtFQVZEO0lBT0ksYUFBWTtJQUNaLGNBQWEsRUFDZDs7QUFJSDtFQUNFLGFBQVk7RUFDWixtQkFBa0I7RUFDbEIsOEJBQTZCO0VBQzdCLGFBQVk7RUFDWixjQUFhLEVBQ2Q7O0FBRUQ7RUFFSSxrQkFBaUI7RUFDakIsY1JwQmlCLEVRcUJsQjs7QUFKSDtFQU1JLG9DQUFtQztFQUduQyw0QkFBMkI7RUFDM0IsMkNBQXlDO0VBRXpDLG1DQUFpQyxFQUNsQzs7QUFJSDtFQUEwQjtJQUFPLGtDQUFpQyxFQUFBLEVBQUE7O0FBQ2xFO0VBQWtCO0lBQU8sa0NBQWlDO0lBQUUsMEJBQXdCLEVBQUEsRUFBQTs7QUNuRHBGO0VBQ0UsOEJBQTZCLEVBRTlCOztBQUVEO0VBQ0UsZ0JBQWU7RUFDZixTQUFRO0VBQ1IsOEJBQTRCLEVBQzdCOztBWE1EOztHQUVHO0FBQ0g7RUFDRTtJQUNFLDRCQUEyQixFQUM1QjtFQUNEO0lBQ0UsMkJBQTBCO0lBQzFCLFVBQVMsRUFDVjtFQUNEO0lBQ0Usa0JBQWlCLEVBQ2xCO0VBRUg7O01BRU0sRUFBQTs7QUFHTjs7R0FFRztBQWdDSDtFQUNFLGdDQUErQjtFQUMvQixnQkFBZSxFQUNoQjs7QUFFRDtFQUNFLG1DQUFrQztFQUNsQyxlRWpFaUIsRUZrRWxCOztBQUVEO0VBQ0UsZ0JBQWU7RUFDZixzQkFBcUI7RUFDckIsZUUvRVksRUZnRmI7O0FBRUQ7RUFDRSxhQUFZLEVBSWI7RUFMRDtJQUdJLHNCQUFxQixFQUN0Qjs7QUFHSDtFQUNFLCtCQUE4QjtFQUM5QiwrQkFBOEIsRUFDL0I7O0FBRUQ7RUFDRSxpQkFBZ0IsRUFDakI7O0FBRUQ7RUFDRSxZQUFXO0VBQ1gsYUFBWTtFQUNaLGtCQUFpQjtFQUNqQixtQkFBa0I7RUFDbEIsVUFBUyxFQUlWO0VBVEQ7SUFPSSxjRTVGaUIsRUY2RmxCOztBQUdIOztJQUVJO0FBRUo7RUFDRSxpQkFBZ0I7RUFDaEIsb0JBQW1CLEVBQ3BCOztBQUNEO0VBQ0UsaUJBQWdCO0VBQ2hCLG9CQUFtQixFQUNwQjs7QUFDRDtFQUNFLGlCQUFnQjtFQUNoQixvQkFBbUIsRUFDcEI7O0FBQ0Q7RUFDRSxpQkFBZ0I7RUFDaEIsb0JBQW1CLEVBQ3BCOztBQUdEO0VBQ0UsaUJBQWdCLEVBQ2pCOztBQUVEO0VBQ0UsYUFBVyxFQUNaOztBQUVEO0VBQ0UsZ0NBQThCLEVBQy9COztBQUVEO0VBQ0UsMEJBQXlCO0VBQ3pCLGdCQUFlO0VBQ2Ysa0JBQWdCO0VBQ2hCLG9CQUFrQjtFQUNsQixpQkFBZ0I7RUFDaEIsa0JBQWlCLEVBWWxCO0VBbEJEO0lBU0ksaUJBQWU7SUFDZixhQUFXO0lBQ1gsWUFBVTtJQUNWLGlCQUFlO0lBQ2YsZ0JBQWMsRUFDZjtFQWRIO0lBZ0JJLGdCQUFjLEVBQ2Y7O0FBRUg7RUFDRSwwQkVyS29CO0VGc0twQixlRTVKaUIsRUZnS2xCO0VBTkQ7SUFJSSxjRTFKVSxFRjJKWDs7QUFFSDtFQUNFLDBCRTNLc0I7RUY0S3RCLGVFbktpQixFRnVLbEI7RUFORDtJQUlJLGNFaEtpQixFRmlLbEI7O0FBR0g7O0VBRUUsZUFBYSxFQUNkOztBQUVEO0VBQ0csVUFBUSxFQUNWOztBQUVEO0VBRUksaUJBQWUsRUFDaEI7O0FBR0g7RUFDRSxlRXpMaUIsRUY4TGxCO0VBTkQ7SUFHSSxzQkFBcUI7SUFDckIsZUVwTVUsRUZxTVg7O0FBR0g7RUFDRSxlRTVNYztFRjZNZCxzQkFBcUI7RUFDckIsd0RBQXVELEVBS3hEO0VBUkQ7SUFLSSxzQkFBcUI7SUFDckIsZUU5TVUsRUYrTVg7O0FBUUg7O0VBRUU7QUFFRjtFQUNFLHFDQUF5RDtFQUN6RCxZRW5MaUI7RUZvTGpCLHdCQUF1QixFQUt4QjtFQVJEO0lBS0ksdUJBQXNCLEVBQ3ZCOztBQUlIO0VBRUUsNEJBQTJCLEVBQzVCOztBQUVEO0VBRUksMENBQXlDLEVBQzFDOztBQUdIO0VBQ0UsZUVqUFk7RUZrUFosVUFBUyxFQUNWOztBQUNEOztFQUVFLGVFblBXLEVGdVBaOztBQUVEO0VBQ0Usa0JBQWlCLEVBQ2xCOztBQUVEO0VBQ0UsY0FBYSxFQW1CZDtFQXBCRDtJQUlJLDBCQUF5QjtJQUN6QixzQkFBcUI7SUFDckIsYUFBWSxFQUNiO0VBUEg7SUFTSSwwQkFBeUI7SUFDekIsVUFBUSxFQU1UO0lBaEJIO01BWU0sY0FBYTtNQUNiLGFBQVk7TUFDWixZQUFXLEVBQ1o7RUFmTDtJQWtCSSxZQUFXLEVBQ1o7O0FBR0g7RUFFSSxhQUFXLEVBQ1o7O0FBSEg7RUFLSSxVQUFTO0VBQ1QsZ0JBQWU7RUFDZixpQkFBZ0I7RUFDaEIsZUVsUlUsRUZtUlg7O0FBR0g7RUFFSSxhQUFXLEVBQ1o7O0FBSEg7RUFLSSxVQUFRLEVBQ1Q7O0FBTkg7RUFRSSxlRTlSVSxFRitSWDs7QUFHSDs7R0FFRztBQUdIO0VBQ0Usd0JBQXVCLEVBQ3hCOztBQUNEOztFQUVFLDBCQUF5QixFQUMxQjs7QUFDRDtFQUNFLDZCQUE0QixFQUM3Qjs7QUFjRDtFQUNFLDBCRTNVWTtFRjRVWixhQUFZLEVBQ2I7O0FBRUQ7RUFDRSwwQkVoVlk7RUZpVlosYUFBWSxFQUNiOztBQUVEO0VBQ0UsdUJBQXNCLEVBQ3ZCOztBQUVEO0VBRUksZUUxVlUsRUYyVlg7O0FBSEg7RUFNSSxrQ0FBOEIsRUFDL0I7O0FBR0g7O0dBRUc7QUFFSDtFQUNFLDZCQUE0QjtFQUM1QixtQ0FBa0M7RUFDbEMsd0JBQXVCO0VBQ3ZCLGdCQUFlLEVBQ2hCOztBQUVEO0VBSUUseUJBQXdCO0VBQ3hCLGFBQVk7RUFDWixjQUFhO0VBQ2IsV0FBVTtFQUNWLGdCQUFlO0VBQ2YscUJBQW9CO0VBQ3BCLGdCQUFlO0VBQ2YsWUFBVyxFQXNCWjtFQWpDRDtJQWNJLFlBQVc7SUFDWCxhQUFZO0lBQ1osbUJBQWtCLEVBQ25CO0VBakJIO0lBb0JJLGNBQWE7SUFDYixlQUFjO0lBQ2QsZUFBYyxFQUNmO0VBdkJIO0lBMEJJLGtCQUFpQjtJQUNqQixpQkFBZ0I7SUFDaEIsd0JBQXVCO0lBQ3ZCLDBCRTdZWTtJRjhZWixvQkFBbUI7SUFDbkIsZ0JBQWUsRUFDaEI7O0FBR0g7RUFDRSwwQkU5WVcsRUYrWVo7O0FBRUQ7RUFDRSwwQkUzWG1CLEVGNFhwQjs7QUFFRDtFQUNFLDBCRXpaWSxFRjBaYjs7QUFFRDtFQUNFLDBCRWhhYyxFRmlhZjs7QUFFRDtFQUNJLDBCRWphVSxFRmthYjs7QUFDRDtFQUNJLHFCQUFvQixFQUN2Qjs7QUFDRDtFQUNJLHFCQUFvQjtFQUNwQiwwQkFBeUI7RUFDekIsdUJBQXNCO0VBQ3RCLHNCQUFxQjtFQUNyQixrQkFBaUIsRUFDcEI7O0FBRUQ7O0dBRUc7QUFFSDtFQUNFLDBCRS9hYSxFRmdiZDs7QUFDRDtFQUNFLHVDRWxiYSxFRm1iZDs7QUFDRDtFQUNFLDBCRXJiYSxFRnNiZDs7QUFDRDtFQUNFLDBCRTViWSxFRjZiYjs7QUFDRDtFQUNFLHlDRS9iWSxFRmdjYjs7QUFDRDtFQUNFLDBCRWxjWSxFRm1jYjs7QUFFRDtFQUNFLGFBQVk7RUFDWixtQkFBa0I7RUFDbEIsUUFBTztFQUNQLFlBQVcsRUFDWjs7QUFFRDtFQUNFLG1CQUFrQjtFQUNsQixXQUFVO0VBQ1YsVUFBUyxFQUNWOztBQUVEOztHQUVHO0FBRUY7RUFDRywwQkFBcUM7RUFDckMsZUUzZFk7RUY0ZFosZ0JBQWU7RUFDZixVQUFTO0VBQ1QsYUFBVztFQUNYLFFBQU87RUFDUCxZQUFXO0VBRVgsYUFBWTtFQUNaLG1CQUFrQjtFQUNsQixpQkFBZ0IsRUFLbEI7RUFoQkQ7SUFjSyw4QkV2ZVUsRUZ3ZVg7O0FBS0w7RUFDRSxZQUFXO0VBQ1gsY0FBa0I7RUFDbEIsbUJBQWtCO0VBQ2xCLHdCQUF1QixFQXVCeEI7RUEzQkQ7SUFPSSxlQUFjLEVBRWY7RUFUSDtJQVdJLG1CQUFrQixFQUNuQjtFQVpIO0lBY0ksZUUzZWlCO0lGNGVqQixpQkFBZ0I7SUFDaEIsaUJBQWdCLEVBQ2pCO0VBakJIO0lBbUJJLFVBQVM7SUFDVCxnQkFBZTtJQUNmLGlCQUFnQixFQUNqQjtFQXRCSDtJQXdCSSxlRWxnQlU7SUZtZ0JWLHNCQUFxQixFQUN0Qjs7QUFHSDs7R0FFRztBQUVIO0VBQ0UsZ0JBQWU7RUFDZixXQUFVLEVBQ1g7O0FBT0Q7RUFDRSx3QkFBdUI7RUFDdkIsbUJBQWtCO0VBQ2xCLFlBQVc7RUFDWCxhQUFZO0VBQ1osYUFBWTtFQUNaLFdBQVU7RUFDVixnQkFBZTtFQUNmLFVBQVMsRUFtQlY7RUFqQkM7SUFWRjtNQVdJLGNBQWEsRUFnQmhCLEVBQUE7RUFkQztJQWJGO01BY0ksY0FBYSxFQWFoQixFQUFBO0VBWEM7SUFoQkY7TUFpQkksY0FBYSxFQVVoQixFQUFBO0VBUkM7SUFuQkY7TUFvQkksY0FBYSxFQU9oQixFQUFBO0VBM0JEO0lBeUJJLGVBQWMsRUFDZjs7QUFHSDs7R0FFRztBQUNIO0VBQ0Usc0JBQXFCLEVBU3RCO0VBVkQ7SUFJSSxlQUFhLEVBQ2Q7RUFMSDtJQVFJLDBDQUFzQyxFQUN2Qzs7QUFFSDtFQUVFLDJCQUEwQjtFQUMxQixpQkFBZ0I7RUFDaEIsb0JBQW1CO0VBQ25CLGFBQVc7RUFDWCwwQkFBeUI7RUFDekIsZ0JBQWU7RUFDZixrQkFBaUIsRUF3QmxCO0VBaENEO0lBV0ksaUJBQWdCO0lBQ2hCLGtCQUFpQixFQUNsQjtFQWJIO0lBZUksVUFBUztJQUNULDBCQUF5QixFQUsxQjtJQXJCSDtNQWtCTSxXQUFVO01BQ1YsWUFBVyxFQUNaO0VBcEJMO0lBd0JJLHdCQUF1QjtJQUN2QiwwQ0FBc0MsRUFDdkM7RUExQkg7SUE0QkksMEJFaGxCVTtJRmlsQlYsWUFBVztJQUNYLFlBQVcsRUFDWjs7QUFFSDs7RUFFRTtBQUNGO0VBRUUsbUJBQWtCO0VBQ2xCLFVBQVM7RUFDVCxrQkFBaUI7RUFDakIsY0FBYSxFQTZDZDtFQWxERDtJQVFJLHFCQUFvQjtJQUNwQixvQkFBbUI7SUFDbkIsa0JBQWlCO0lBQ2pCLGNBQWEsRUFDZDtFQVpIO0lBZUksY0FBYTtJQUNiLHVCQUFzQjtJQUN0QiwrQkFBOEI7SUFDOUIsYUFBWTtJQUNaLGFBQVk7SUFFWixtQkFBaUI7SUFDakIsaUJBQWU7SUFDZiwwQ0FBeUMsRUEwQjFDO0lBakRIO01BeUJNLGNFcm5CYSxFRnNuQmQ7SUExQkw7TUE4Qk0sZ0JBQWU7TUFNZixjQUFhO01BRWIsb0JBQW1CO01BQ25CLGtCQUFpQjtNQUNqQix1QkFBc0IsRUFRdkI7TUFoREw7UUFpQ1EsOEJBQTZCLEVBQzlCO01BbENQO1FBMkNRLGlCQUFlLEVBSWhCO1FBL0NQO1VBNkNVLFVBQVMsRUFDVjs7QUFLVDtFQUNFLHdCQUF1QjtFQUN2QixtQkFBa0I7RUFDbEIsWUFBVztFQUNYLGFBQVk7RUFDWiwwQ0FBc0MsRUFNdkM7RUFYRDtJQVFJLFlBQVc7SUFDWCxhQUFZLEVBQ2I7O0FBRUg7RUFDRSx3QkFBdUIsRUFDeEI7O0FBSUQ7Ozs7RUFJRTtBQUVGO0VBQ0Usa0JBQWlCLEVBc0JsQjtFQXZCRDtJQUlJLDBCRXhxQmtCO0lGeXFCbEIsY0FBYSxFQUNkO0VBTkg7SUFTSSxxQkFBb0IsRUFhckI7SUF0Qkg7TUFZTSxjQUFhO01BQ2IsK0JBQThCO01BQzlCLGtCQUFpQjtNQUNqQixtQkFBa0IsRUFNbkI7TUFyQkw7UUFpQlEsZ0JBQWM7UUFDZCxhQUFXO1FBQ1gsaUJBQWdCLEVBQ2pCOztBQUtQOztHQUVHO0FBRUg7RUFDRSxnQkFBZTtFQUNmLG1CQUFrQjtFQUNsQiwrQkFBOEIsRUFDL0I7O0FBQ0Q7RUFDRSxXQUFVO0VBQ1YsZUFBYyxFQUNmOztBQUNEO0VBQ0UsbUJBQWtCLEVBQ25COztBQUVEO0VBQ0UsbUJBQWtCO0VBQ2xCLFlBQVc7RUFDWCxTQUFRO0VBQ1IsV0FBVTtFQUNWLGlCQUFnQjtFQUNoQixlQUFjLEVBQ2Y7O0FBRUQ7RUFDRSxhQUFZO0VBQ1osWUFBVztFQUNYLGFBQVk7RUFDWixhQUFZLEVBU2I7RUFiRDtJQU9JLFlBQVc7SUFDWCxhQUFZO0lBQ1osbUJBQWtCO0lBQ2xCLFNBQVE7SUFDUixVQUFTLEVBQ1Y7O0FBR0g7RUFDRSxhRXR0QnNCO0VGdXRCdEIsMEJFbnZCWTtFRm92QlosYUFBWTtFQUNaLFlBQVc7RUFDWCxVQUFTO0VBQ1QsK0JBQThCLEVBQy9COztBQUdEO0VBQ0UsZUU1dkJZO0VGNnZCWixvQkFBbUIsRUFDcEI7O0FBRUQ7RUFDRSxpQkFBZ0I7RUFDaEIsZUVydkJtQjtFRnN2Qm5CLGtCQUFpQjtFQUNqQixlQUFjO0VBQ2QsaUJBQWdCLEVBQ2pCOztBQUNEO0VBQ0UsbUJBQWtCO0VBQ2xCLDhCQUEwQjtFQUMxQixpQkFBZ0IsRUFDakI7O0FBRUQ7RUFDRSxnQkFBZTtFQUNmLGlCQUFnQixFQUNqQjs7QUFFRDs7RUFFRTtBQUNGO0VBQ0U7SUFDRSxhQUFZO0lBQ1osZ0JBQWUsRUFDaEI7RUFDRDtJQUNFLG9CQUFtQixFQUNwQjtFQUNEO0lBQ0UsYUFBWTtJQUNaLGVBQWM7SUFDZCxtQkFBa0I7SUFDbEIsZUFBYztJQUNkLGlCQUFnQixFQUNqQixFQUFBOztBQUdIOztFQUVFO0FBRUY7RUFDRTtJQUNFLFdBQVU7SUFDVixnQkFBZSxFQUNoQjtFQUNEO0lBQ0UsWUFBVztJQUNYLG1CQUFrQjtJQUNsQixpQkFBZ0IsRUFDakIsRUFBQTs7QUFJSDtFQUNFO0lBQ0UsaUJBQWdCLEVBQ2pCO0VBQ0Q7SUFDRSxjQUFhLEVBQ2QsRUFBQTs7QUFHSDs7RUFFRTtBQUNGO0VBQ0UsbUJBQWtCO0VBQ2xCLGlCQUFnQjtFQUNoQixxQkFBbUIsRUFFcEI7O0FBRUQ7RUFDRSxlQUFjLEVBQ2Y7O0FBRUQ7RUFDRTtJQUNFLGlCQUFnQixFQUNqQjtFQUNEO0lBQ0UsWUFBVztJQUNYLFNBQVEsRUFDVCxFQUFBOztBQUdIO0VBQ0U7SUFDRSxnQkFBZSxFQUNoQjtFQUNEO0lBQ0UsWUFBVztJQUNYLFNBQVEsRUFDVCxFQUFBOztBQUtEO0VBQ0U7SUFDRSxjQUFhLEVBQ2Q7RUFDRDtJQUNFLFlBQVcsRUFDWixFQUFBOztBQUlMOztFQUVFO0FBQ0Y7RUFDRSxjQUFhLEVBQ2Q7O0FBRUQ7RUFDRTtJQUNFLFdBQVU7SUFDVixTQUFRLEVBQ1QsRUFBQTs7QUFHSDtFQUNFO0lBQ0UsWUFBVztJQUNYLFNBQVEsRUFDVCxFQUFBOztBQUlEO0VBQ0U7SUFDRSxTQUFRLEVBQ1Q7RUFDRDtJQUNFLGNBQWEsRUFDZCxFQUFBOztBQUlMOztHQUVHO0FBRUg7RUFFRSxvQ0FBbUM7RUFHbkMsNEJBQTJCLEVBQzVCOztBQUVEOztFQUVFO0FBRUY7RUFDRSwwQkVsNkJjO0VGbTZCZCxvQkFBbUI7RUFDbkIsK0NBQTRDLEVBQzdDOztBQUVEO0VBQ0UsMEJFeDZCYztFRnk2QmQsb0JBQW1CO0VBQ25CLCtDQUE0QztFQUM1Qyx5Q0UzNkJjLEVGNDZCZjs7QUFFRDtFQUNFLDBCRWg2Qlk7RUZpNkJaLG9CQUFtQjtFQUNuQiwrQ0FBNEM7RUFDNUMsMEJFbjZCWSxFRm82QmI7O0FBRUQ7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7SUEyQ0k7QUFFSjtFQUNFLHFDQUFrQztFQUNsQyxvQkFBbUI7RUFDbkIsK0NBQTRDO0VBQzVDLHFDQUFpQyxFQUNsQzs7QUFFRDs7Ozs7SUFLSTtBQUVKO0VBQ0UsWUFBVztFQUNYLGNBQWE7RUFDYixlRXgrQmlCO0VGeStCakIsZ0NBQ0YsRUFBQzs7QUFFRDtFQUNFLGFBQVk7RUFDWiwyQkFBMEI7RUFDMUIsMEJFNStCb0I7RUY2K0JwQixpQkFBZSxFQTZCaEI7RUFqQ0Q7SUFPSSxlRW4vQmUsRUZ1L0JoQjtJQVhIO01BU00sZUVoL0JlLEVGaS9CaEI7RUFWTDtJQWNJLHFCQUFvQjtJQUNwQixnQkFBZTtJQUNmLGtCQUFpQixFQUNsQjtFQWpCSDtJQW1CSSxnQkFBZTtJQUNmLGlCQUFnQjtJQUNoQix1QkFBc0IsRUFDdkI7RUF0Qkg7SUF3QkksZ0JBQWU7SUFDZixxQkFBb0IsRUFPckI7SUFoQ0g7TUE0Qk0sWUFBVztNQUNYLGFBQVk7TUFDWixrQkFBaUIsRUFDbEI7O0FBSUw7RUFDRSxhQUFZLEVBNEJiO0VBN0JEO0lBSUkscUJBQW9CO0lBQ3BCLGdCQUFlLEVBT2hCO0lBWkg7TUFRTSxhQUFZO01BQ1osYUFBWTtNQUNaLGtCQUFpQixFQUNsQjtFQVhMO0lBY0kscUJBQW9CO0lBQ3BCLGdCQUFlLEVBYWhCO0lBNUJIO01BeUJNLG1CQUFrQjtNQUNsQixZQUFXLEVBQ1o7O0FBR0w7RUFDRSxTQUFRO0VBQ1IsVUFBUyxFQUNWOztBQUVELHNEQUFzRDtBQUV0RDtFQUNFLDBCRWhrQ2MsRUZpa0NmOztBQUNEO0VBQ0UsdUJBQXNCLEVBQ3ZCOztBQUNEOztHQUVHO0FBRUg7RUFHRSx1QkFBc0I7RUFDdEIsWUFBVztFQUNYLHFCQUFvQixFQUNyQjs7QUFFRDtFQUNFLG1CQUFrQjtFQUNsQix1QkFBc0I7RUFDdEIsaUJBQWdCLEVBQ2pCOztBQUVEO0VBQ0UsVUFBUztFQUNULGlCQUFnQjtFQUNoQixtQkFBa0I7RUFDbEIsVUFBUztFQUNULFdBQVUsRUFLWDtFQVZEO0lBUUksVUFBUyxFQUNWOztBQUdIO0VBQ0UsY0FBYSxFQUNkOztBQUVEO0VBQ0UsYUFBWSxFQUNiOztBQUVEO0VBQ0UsV0FBVSxFQUNYOztBQUNEO0VBQ0UsV0FBVSxFQUNYOztBQUVEO0VBQ0UsK0NBQTRDLEVBVzdDO0VBWkQ7SUFPSSxnQ0FBK0I7SUFDL0IsZ0JBQWUsRUFHaEI7O0FBR0g7RUFDSSxtQkFBa0I7RUFDbEIseUNFbG9DWTtFRm1vQ1oseUNFbm9DWSxFRnVvQ2Y7RUFQRDtJQUtJLHVDQUFtQyxFQUNwQzs7QUFHSDs7RUFFRTtBQUVGO0VBQ0UsZUVub0NpQixFRm9vQ2xCOztBQUVEO0VBQ0UsZUVsb0NtQixFRm1vQ3BCOztBQUVEO0VBQ0UsbUNFdG9DbUIsRUZ1b0NwQjs7QUFFRDtFQUNFLGtDRTFvQ21CLEVGMm9DcEI7O0FBRUQ7RUFDSSxpQ0UzcENVLEVGNHBDYjs7QUFFRDtFQUNFLGlCQUFnQjtFQUNoQixpQkFBZ0IsRUFDakI7O0FBRUQ7RUFDSSw4QkVqcUNTLEVGa3FDWjs7QUFDRDtFQUNFLGVFL3BDaUIsRUZncUNsQjs7QUFDRDtFQUNFLGVFbHFDaUIsRUZtcUNsQjs7QUFDRDtFQUNFLG9CRWhyQ2M7RUZpckNkLDBCRWpyQ2M7RUZrckNkLHNCRWxyQ2M7RUZtckNkLGVFeHFDaUIsRUZ5cUNsQjs7QUFDRDtFQUNFLG9CRXByQ3NCO0VGcXJDdEIsc0JFcnJDc0IsRUZzckN2Qjs7QUFFRDtFQUNFLGFBQVk7RUFDWixtQkFBa0I7RUFDbEIsMEJFN3FDbUI7RUY4cUNuQixlRTlxQ21CO0VGK3FDbkIsOEJBQTZCLEVBQzlCOztBQUVEO0VBQ0UsbUJBQWtCLEVBQ25COztBQUVEOzs7R0FHRztBQUNGO0VBQ0UsZUUzckNrQixFRjRyQ25COztBQUVGO0VBR0ksWUFBVztFQUNYLHNCQUFxQjtFQUNyQixhQUFZLEVBQ2I7O0FBTkg7RUFRSSxTQUFRO0VBQ1IsaUJBQWdCO0VBQ2hCLFlBQVc7RUFDWCxhQUFZLEVBS2I7RUFoQkg7SUFhTSxtQkFBa0I7SUFDbEIsbUJBQWtCLEVBQ25COztBQUdMO0VBR0ksWUFBVztFQUNYLHNCQUFxQixFQUN0Qjs7QUFMSDtFQU9JLFNBQVE7RUFDUixpQkFBZ0I7RUFDaEIsWUFBVztFQUNYLGFBQVk7RUFDWixtQkFBa0IsRUFLbkI7RUFoQkg7SUFhTSxtQkFBa0I7SUFDbEIsbUJBQWtCLEVBQ25COztBQUlMO0VBQ0UsZUFBYztFQUNkLGVBQWMsRUFDZjs7QUFFRDs7O0dBR0c7QUFDSDtFQUNFLFlBQVc7RUFDWCxlRS91Q1k7RUZndkNaLDhCQUE2QjtFQUM3QixnQ0FBK0I7RUFDL0IsK0JBQThCO0VBQzlCLHdCQUF1QjtFQUN2Qiw0QkFBMkI7RUFDM0IscUJBQW9CO0VBQ3BCLGNBQVksRUE4QmI7RUF2Q0Q7SUFZSSxxQkFBb0IsRUFDckI7RUFiSDtJQWVJLG1CQUFrQixFQUNuQjtFQWhCSDtJQW1CSSxZQUFXO0lBQ1gsc0JBQXFCLEVBQ3RCO0VBckJIO0lBdUJJLFNBQVE7SUFDUixpQkFBZ0I7SUFDaEIsWUFBVztJQUNYLGFBQVk7SUFDWixtQkFBa0IsRUFLbkI7SUFoQ0g7TUE2Qk0sbUJBQWtCO01BQ2xCLG1CQUFrQixFQUNuQjtFQS9CTDtJQW1DSSxtQkFBa0I7SUFDbEIsYUFBWSxFQUNiOztBQUlIO0VBQ0UsWUFBVztFQUNYLGFBQVk7RUFDWixvQkFBbUIsRUFDcEI7O0FBQ0Q7RUFDRSxtQkFBa0I7RUFDbEIsZ0JBQWU7RUFDZixlRTF5Q1ksRUYyeUNiOztBQUNEO0VBQ0UsbUJBQWtCO0VBQ2xCLFNBQVEsRUFLVDtFQVBEO0lBS0ksMEJBQXlCLEVBQzFCOztBQUVIO0VBQ0UsYUFBVztFQUNYLDBCQUF5QixFQUMxQjs7QUFDRDtFQUNFLGVFcnpDYSxFRnN6Q2Q7O0FBQ0Q7RUFDRSxlRS96Q2MsRUZnMENmOztBQUNEO0VBQ0UsZUFBYyxFQUNmOztBQUNEO0VBQ0UsZUFBYztFQUNkLGVBQWMsRUFLZjtFQVBEO0lBS0ksbUJBQWtCLEVBQ25COztBQUdIO0VBQ0UsZUFBYyxFQUNmOztBQUVEO0VBQ0ksbUJBQWtCO0VBQ2xCLG1CQUFrQjtFQUNsQixZQUFXLEVBQ2Q7O0FBQ0Q7RUFDRSxxQ0FBb0MsRUFDckM7O0FBRUQ7RUFDRSxvQkFBbUIsRUFDcEI7O0FBRUQ7OztHQUdHO0FBRUgsb0JBQW9CO0FBQ3BCO0VBQ0UsMEJFMzFDeUI7RUY0MUN6QixlRXYxQ1k7RUZ3MUNaLGNBQWE7RUFDYixpQkFBZ0IsRUFDakI7O0FBQ0Q7RUFDRSxhQUFZO0VBQ1osY0FBYTtFQUNiLG9CQUFtQjtFQUNuQixtQkFBa0I7RUFDbEIsa0JBQWlCLEVBQ2xCOztBQUNEO0VBRUksY0FBYSxFQUNkOztBQUhIO0VBS0ksY0V4MkNrQixFRnkyQ25COztBQUVIO0VBQ0UsWUFBVztFQUNYLGFBQVk7RUFDWixrQkFBaUI7RUFDakIsbUJBQWtCO0VBQ2xCLFVBQVMsRUFDVjs7QUFFRDtFQUNFLGtCQUFpQjtFQUNqQixtQkFBa0IsRUFDbkI7O0FBQ0QscUJBQXFCO0FBQ3JCO0VBSUUsMEJBQXlCO0VBQ3pCLGlCQUFlLEVBQ2hCOztBQUNEO0VBQ0UsZUVuNENpQjtFRm80Q2pCLGtCQUFpQixFQUNsQjs7QUFDRDtFQUNFLGdCQUFlLEVBQ2hCOztBQUVEO0VBQ0UsYUFBWTtFQUNaLGVFcDVDWSxFRnM1Q2I7RUFKRDtJQUdRLGNFcjVDTSxFRnE1Q1M7O0FBR3ZCO0VBQ0UsWUFBVztFQUNYLGFBQVk7RUFDWixrQkFBaUI7RUFDakIsbUJBQWtCO0VBQ2xCLFVBQVMsRUFDVjs7QUFFRCwwQkFBMEI7QUFDMUI7RUFDRSxzQkFBcUIsRUFDdEI7O0FBRUQ7O0dBRUc7QUFHSDtFQUdJLDBCRTc2Q1UsRUY4NkNYOztBQUpIO0VBTUksa0JBQWlCLEVBQ2xCOztBQVBIO0VBU0ksZUV2NkNVLEVGdzZDWDs7QUFWSDtFQVlJLGFBQVksRUFDYjs7QUFiSDtFQWVJLGVFejdDVTtFRjA3Q1YsMEJFMTdDVSxFRjI3Q1g7O0FBR0g7RUFDRSwwQkFBeUIsRUFDMUI7O0FBRUQ7RUFDRSxjQUFhO0VBQ2IsbUJBQWtCLEVBQ25COztBQUNEO0VBQ0Usa0JBQWlCO0VBQ2pCLGtCQUFpQixFQUNsQjs7QUFDRDtFQUNFLGVBQWMsRUFDZjs7QUFDRDtFQUVJLDBCRXY4Q2UsRUZ3OENoQjs7QUFJSDtFQUNFLFlBQVc7RUFDWCxhQUFZO0VBQ1osa0JBQWlCO0VBQ2pCLG1CQUFrQjtFQUNsQixVQUFTLEVBSVY7RUFURDtJQU9JLFlBQVcsRUFDWjs7QUFFSDtFQUNFLG1CQUFrQixFQUNuQjs7QUFDRDtFQUNFLG1CQUFrQjtFQUNsQixpQkFBZ0IsRUFDakI7O0FBQ0Q7RUFDRSxlQUFjO0VBQ2QsbUJBQWtCLEVBQ25COztBQUNEO0VBRUksOEJBQTZCLEVBQzlCOztBQUhIO0VBS0ksNEJBQTJCLEVBQzVCOztBQU5IO0VBUUksZ0JBQWUsRUFDaEI7O0FBVEg7RUFXSSxtQkFBa0I7RUFDbEIsYUFBWSxFQUNiOztBQUVIO0VBQ0Usb0JBQW1CLEVBQ3BCOztBQUNEO0VBQ0UsWUFBVztFQUNYLGFBQVk7RUFDWixvQkFBbUI7RUFDbkIsaUJBQWdCLEVBQ2pCOztBQUNEO0VBQ0UsZUV0L0NZO0VGdS9DWixrQkFBaUI7RUFDakIsYUFBWSxFQUNiOztBQUNEO0VBQ0UsZUFBYztFQUNkLGtCQUFpQjtFQUNqQiwwQkFBeUI7RUFDekIsa0JBQWlCLEVBQ2xCOztBQUNEO0VBQ0UsMEJFamdEWTtFRmtnRFosYUFBWSxFQVFiO0VBVkQ7SUFLSSxhQUFZO0lBQ1osWUFBVztJQUNYLGVBQWM7SUFDZCxpQkFBZ0IsRUFDakI7O0FBRUg7RUFDRSwyQkFBMEIsRUFDM0I7O0FBQ0Q7RUFDRSwwQkUvZ0RZLEVGZ2hEYjs7QUFDRDtFQUNFLGVBQWM7RUFDZCxlRTVoRFc7RUY2aERYLG1CQUFrQixFQUNuQjs7QUFDRDtFQUNFLG1CQUFrQjtFQUNsQixVQUFTO0VBQ1QsVUFBUztFQUNULFdBQVU7RUFDVixnQkFBZTtFQUNmLGVBQWMsRUFDZjs7QUFDRDtFQUNFLGlCQUFnQjtFQUNoQixvQkFBbUIsRUFDcEI7O0FBRUQ7RUFDRSxpQkFBZ0I7RUFDaEIsb0JBQW1CLEVBSXBCO0VBTkQ7SUFJSSxjRXRpRGlCLEVGdWlEbEI7O0FBR0g7RUFDRSwwQkV4akRZLEVGeWpEYjs7QUFDRDtFQUNFLDBCRTNqRFksRUY0akRiOztBQUNEO0VBQ0UsbUJBQWlCO0VBQ2pCLGFBQVk7RUFDWixZQUFVO0VBQ1YsT0FBSztFQUNMLFFBQU0sRUFDUDs7QUFFRDs7O0dBR0c7QUFFRjtFQUVJLGVFcGtEYyxFRnFrRGY7O0FBSEg7RUFLSSxlRWxrRGdCLEVGbWtEakI7O0FBR0o7RUFDRSxZQUFXLEVBQ1o7O0FBRUQ7O0VBRUU7QUFFRjtFQUVFLFdBQVU7RUFDVixZQUFXO0VBQ1gsMEJFdmxEaUIsRUZ3bERsQjs7QUFFRDtFQUNFLGVBQWM7RUFDZCxZQUFXO0VBQ1gsYUFBWSxFQUNiOztBQUNEO0VBQ0Usb0JBQW1CLEVBQ3BCOztBQUNEO0VBQ0Usd0JBQXVCLEVBQ3hCOztBQUNEO0VBQ0UsYUFBWTtFQUNaLG9CQUFtQjtFQUNuQix1QkFBc0I7RUFDdEIsa0JBQWlCLEVBQ2xCOztBQUNEO0VBQ0UsY0FBYTtFQUNiLHVCQUFzQjtFQUN0Qix1QkFBc0I7RUFDdEIsd0JBQXVCLEVBQ3hCOztBQUNEO0VBQ0UsMkJBQTBCO0VBQzFCLFlBQVcsRUFDWjs7QUFDRDtFQUNFO0lBQ0UsbUJBQWtCLEVBQ25CO0VBQ0Q7SUFDRSxtQkFBa0IsRUFDbkI7RUFDRDtJQUNFLCtCQUE4QixFQUMvQjtFQUNEO0lBQ0Usd0JBQXVCO0lBQ3ZCLG9CQUFtQixFQUNwQixFQUFBOztBQUVIO0VBQ0U7SUFDRSxvQkFBbUIsRUFDcEIsRUFBQTs7QUFJSDs7RUFFRTtBQUNGO0VBQ0UsWUFBVztFQUNYLGNBQWE7RUFDYiwwQkVqcERpQixFRmtwRGxCOztBQUVEO0VBQ0UsZUFBYztFQUNkLFlBQVc7RUFDWCxhQUFZLEVBQ2I7O0FBRUQ7Ozs7RUFJRTtBQUVGO0VBR0UsY0V6bkRxQjtFRjBuRHJCLDBCRXJxRHlCO0VGc3FEekIsZ0JBQWU7RUFDZixZQUFXO0VBQ1gsWUFBVztFQUNYLFVBQVM7RUFDVCxtQ0UxcUR5QjtFRjJxRHpCLGNBQWEsRUFDZDs7QUFFRDtFQUNFO0lBQ0UsY0FBYSxFQUNkLEVBQUE7O0FBR0g7RUFDRTtJQUNFLFVBQVMsRUFDVjtFQUNEO0lBQ0UsaUJBQWdCLEVBQ2pCLEVBQUE7O0FBR0g7RUFDRTtJQUNFLFVBQVMsRUFDVjtFQUNEO0lBQ0UsaUJBQWdCLEVBQ2pCLEVBQUE7O0FBR0g7RUFDRSxZQUFXLEVBQ1o7O0FBQ0Q7RUFDRSxhQUFZO0VBQ1osMEJFcnREYztFRnN0RGQsbUJBQWtCO0VBQ2xCLFVBQVMsRUFDVjs7QUFDRDtFQUNFLHdCQUF1QjtFQUN2QixpQkFBZ0I7RUFDaEIsb0JBQWtCLEVBQ25COztBQUVEO0VBQ0UsMEJBQXlCO0VBQ3pCLGVFanVEYztFRmt1RGQsZ0JBQWM7RUFDZCxpQkFBZ0IsRUFLakI7RUFURDtJQU9JLGlCQUFnQixFQUNqQjs7QUFHSDtFQUNFLDBCQUF5QjtFQUN6QixlRTV1RGM7RUY2dURkLGdCQUFjLEVBS2Y7RUFSRDtJQU1JLGlCQUFnQixFQUNqQjs7QUFHSDtFQUNFLGtCQUFpQjtFQUNqQixZQUFXO0VBQ1gsa0JFanNEd0IsRUZrc0R6Qjs7QUF3Q0Q7RUFDRSxrQkFBaUI7RUFDakIsbUJBQWtCLEVBQ25COztBQUVEO0VBQ0UsbUJBQWtCLEVBQ25COztBQUlEOztFQUVFO0FBRUY7RUFFSSxtQkFBa0I7RUFDbEIsdUJBQXNCLEVBV3ZCO0VBZEg7SUFNTSwwQkFBeUIsRUFDMUI7O0FBUEw7RUFnQkksZ0JBQWUsRUFFaEI7O0FBbEJIO0VBb0JJLGdCQUFlLEVBQ2hCOztBQXJCSDtFQXVCSSxnQkFBYyxFQUNmOztBQXhCSDtFQTBCSSxjQUFhO0VBQ2IsaUJBQWdCLEVBQ2pCOztBQTVCSDtFQThCSSxlRTEwRFU7RUYyMERWLG1CQUFrQixFQUNuQjs7QUFoQ0g7RUFrQ0ksaUJBQWdCLEVBQ2pCOztBQW5DSDtFQXFDSSxpQkFBZ0I7RUFDaEIsWUFBVyxFQVVaO0VBaERIO0lBd0NNLFlBQVcsRUFDWjtFQXpDTDtJQTJDTSxtQkFBa0I7SUFDbEIsYUFBWTtJQUNaLGNBQWE7SUFDYixpQkFBZ0IsRUFDakI7O0FBL0NMO0VBa0RJLG1CQUFrQjtFQUNsQixTQUFRLEVBQ1Q7O0FBcERIO0VBc0RJLGtCQUFpQjtFQUNqQixhQUFZLEVBYWI7RUFwRUg7SUF5RE0sdUJBQXNCLEVBQ3ZCO0VBMURMO0lBNERNLGNBQWE7SUFDYixtQkFBa0IsRUFDbkI7RUE5REw7SUFnRU0sZ0JBQWU7SUFDZixlQUFjO0lBQ2QsaUJBQWdCLEVBQ2pCOztBQW5FTDtFQXNFSSxtQkFBa0I7RUFDbEIsWUFBVztFQUNYLFFBQU87RUFDUCxXQUFVO0VBQ1YsaUJBQWdCO0VBQ2hCLGVBQWMsRUFDZjs7QUFHSDtFQUNFLGlCQUFlLEVBQ2hCOztBQUVEOztFQUVFO0FBQ0Y7RUFFSSxnQ0FBK0IsRUFLaEM7RUFQSDtJQUtNLGVBQWEsRUFDZDs7QUFOTDtFQVVJLGdCQUFlO0VBQ2YsZUFBYztFQUNkLG9CQUFtQjtFQUNuQixtQkFBaUIsRUFDbEI7O0FBZEg7RUFpQkksWUFBVztFQUNYLGFBQVk7RUFDWixrQkFBaUI7RUFDakIsbUJBQWtCO0VBQ2xCLG1CQUFrQixFQUNuQjs7QUF0Qkg7RUF5Qk0sbUJBQWtCO0VBQ2xCLHVCQUFzQjtFQUN0QixVQUFTO0VBQ1QsaUJBQWdCO0VBQ2hCLGdCQUFlLEVBUWxCO0VBckNIO0lBK0JRLG1CQUFrQjtJQUNsQixPQUFNO0lBQ04sUUFBTztJQUNQLFlBQVc7SUFDWCxhQUFZLEVBQ2I7O0FBcENQO0VBd0NNLG1CQUFrQjtFQUNsQixpQkFBZ0IsRUFDbkI7O0FBMUNIO0VBNkNJLGtCQUFpQixFQUtsQjtFQWxESDtJQWdETSwwQkFBeUIsRUFDMUI7O0FBakRMO0VBb0RJLGFBQVcsRUFDWjs7QUFyREg7RUF1REksa0JBQWlCO0VBQ2pCLHFCQUFvQixFQUNyQjs7QUF6REg7RUE0REksa0JBQWlCO0VBQ2pCLGFBQVk7RUFDWixvQkFBbUI7RUFDbkIsZ0JBQWU7RUFDZiwwQkVsOERVO0VGbThEVixhQUFXLEVBU1o7RUExRUg7SUFvRU0sMEJBQXlCLEVBQzFCO0VBckVMO0lBd0VNLGlCQUFnQixFQUNqQjs7QUF6RUw7RUE2RUksZUUvOERVLEVGZzlEWDs7QUFHSDtFQUNFLGFBQVksRUFDYjs7QUFFRDtFQUNFO0lBQ0UsZUFBYyxFQUNmLEVBQUE7O0FBS0g7RUFDRSxpQkFBZ0IsRUFDakI7O0FBRUQ7RUFDRSxVQUFTLEVBQ1Y7O0FBRUQ7RUFDRSxVQUFTLEVBQ1Y7O0FBRUQ7RUFDRSxvQkFBbUIsRUFDcEI7O0FBRUQ7RUFDRSxpQkFBZ0IsRUFDakI7O0FBSUQ7RUFDQyxxQ0FBb0MsRUFDcEM7O0FBSUQ7RUFFSSwwQkFBeUIsRUFDMUI7O0FBSEg7RUFNSSwyQ0FBMEMsRUFDM0M7O0FBUEg7RUFVSSw0Q0FBd0MsRUFDekM7O0FBWEg7RUFjSSxpQkFBZ0IsRUFDakI7O0FBZkg7RUFrQkksMEJFcmdFZSxFRmdpRWhCO0VBN0NIO0lBcUJNLGlCQUFnQjtJQUNoQixZQUFXLEVBQ1o7RUF2Qkw7SUEwQk0sWUFBVyxFQUNaO0VBM0JMO0lBOEJNLGVFN2dFUSxFRjhnRVQ7RUEvQkw7SUFrQ00sZUFBYyxFQUNmO0VBbkNMO0lBc0NNLGVBQWM7SUFDZCxpQkFBZ0IsRUFDakI7RUF4Q0w7SUEyQ00sZUV6aEVlLEVGMGhFaEI7O0FBNUNMO0VBZ0RJLGNBQWEsRUFDZDs7QUFqREg7RUFxRE0sbUJBQWtCLEVBQ25COztBQXRETDtFQXlETSwwQkV2aUVlLEVGMmlFaEI7RUE3REw7SUEyRFEsWUFBVyxFQUNaOztBQTVEUDtFQWdFTSwwQkUzakVRLEVGK2pFVDtFQXBFTDtJQWtFUSxZQUFXLEVBQ1o7O0FBbkVQO0VBdUVNLDBCQUF5QixFQUkxQjtFQTNFTDtJQXlFUSxZQUFXLEVBQ1o7O0FBMUVQO0VBOEVNLDBCQUF5QixFQUkxQjtFQWxGTDtJQWdGUSxlQUFjLEVBQ2Y7O0FBakZQO0VBcUZNLHVCQUFzQjtFQUN0QiwwQkFBd0IsRUFJekI7RUExRkw7SUF3RlEsZUFBYyxFQUNmOztBQXpGUDtFQTZGTSxZQUFXO0VBQ1gsYUFBWTtFQUNaLG9CQUFtQjtFQUNuQixZQUFXO0VBQ1gsYUFBWSxFQUNiOztBQWxHTDtFQXFHSyxvQkFBbUIsRUFDbkI7O0FBdEdMO0VBeUdLLG9CQUFtQixFQUNuQjs7QUExR0w7RUE2R0ssb0JBQW1CLEVBQ25COztBQTlHTDtFQWlISyxvQkFBbUIsRUFDbkI7O0FBbEhMO0VBcUhLLG9CQUFtQixFQUNuQjs7QUF0SEw7RUF5SEssb0JBQW1CLEVBQ25COztBQTFITDtFQTZISyxvQkFBbUIsRUFDbkI7O0FBOUhMO0VBaUlLLG9CQUFtQixFQUNuQjs7QUFsSUw7RUFxSUssb0JBQW1CLEVBQ25COztBQXRJTDtFQXlJSyxvQkFBbUIsRUFDbkI7O0FBTUwsdUJBQXVCO0FBRXZCO0VBQ0UsYUFBWTtFQUNaLGtEQUF5QztVQUF6QywwQ0FBeUMsRUFBRTs7QUFFN0M7RUFDRTtJQUNFLHdCQUFvQjtJQUNwQixxRUFFd0IsRUFBQTtFQUMxQjtJQUNFLGFBQVk7SUFDWixxRUFFd0IsRUFBQTtFQUMxQjtJQUNFLDREQUV3QixFQUFBO0VBQzFCO0lBQ0UsbURFanFFVSxFQUFBLEVBQUE7O0FGaXBFZDtFQUNFO0lBQ0Usd0JBQW9CO0lBQ3BCLHFFQUV3QixFQUFBO0VBQzFCO0lBQ0UsYUFBWTtJQUNaLHFFQUV3QixFQUFBO0VBQzFCO0lBQ0UsNERBRXdCLEVBQUE7RUFDMUI7SUFDRSxtREVqcUVVLEVBQUEsRUFBQTs7QUZzcUVkO3VDQUN1QyIsImZpbGUiOiJpbmRleC5jc3MiLCJzb3VyY2VzQ29udGVudCI6WyJAaW1wb3J0IHVybChodHRwczovL2ZvbnRzLmdvb2dsZWFwaXMuY29tL2Nzcz9mYW1pbHk9Um9ib3RvK0NvbmRlbnNlZDo0MDAsMzAwLDMwMGl0YWxpYyw0MDBpdGFsaWMsNzAwLDcwMGl0YWxpY3xSb2JvdG86NDAwLDcwMCw3MDBpdGFsaWMsNDAwaXRhbGljKTtcbkBpbXBvcnQgdXJsKCdodHRwczovL2ZvbnRzLmdvb2dsZWFwaXMuY29tL2Nzcz9mYW1pbHk9S2FuaXQ6NDAwLDUwMCw2MDAsNzAwLDkwMCcpO1xuXG5AaW1wb3J0ICdfdmFyaWFibGVzJztcbkBpbXBvcnQgJ2hlbHBlcnMnO1xuQGltcG9ydCAnY29tcG9uZW50cy9sYW5kaW5nL2xhbmRpbmcnO1xuQGltcG9ydCAnY29tcG9uZW50cy9sYXlvdXQvbGF5b3V0JztcbkBpbXBvcnQgJ2NvbXBvbmVudHMvc2VhcmNoL3NlYXJjaCc7XG5AaW1wb3J0ICdjb21wb25lbnRzL2tpdC9raXQnO1xuQGltcG9ydCAnY29tcG9uZW50cy9raXQvc2hvd0tpdC9zaG93S2l0JztcbkBpbXBvcnQgJ2NvbXBvbmVudHMvc3RvcmUvc3RvcmVNb2RhbCc7XG5AaW1wb3J0ICdjb21wb25lbnRzL3VwbG9hZC9jc3ZVcGxvYWQnO1xuQGltcG9ydCAnY29yZS9hbmltYXRpb24vYmFja2Ryb3AvbG9hZGluZ0JhY2tkcm9wJztcbkBpbXBvcnQgJ2NvcmUvYW5pbWF0aW9uL2FuaW1hdGlvbi5kaXJlY3RpdmUnO1xuXG4vKipcbiAqICBNT1pJTExBLVNQRUNJRklDIENTU1xuICovXG5ALW1vei1kb2N1bWVudCB1cmwtcHJlZml4KCkge1xuICAua2l0X2RldGFpbGVkIC5raXRfY2hhcnQgLmNvbnRhaW5lciAua2l0X2NoYXJ0X2xlZnQgLnNlbnNvcl9zZWxlY3QgLnNlbnNvcl9pY29uX3NlbGVjdGVkIHtcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkoLThweCk7XG4gIH1cbiAgLmtpdF9kZXRhaWxlZCAua2l0X2NoYXJ0IC5jb250YWluZXIgLmtpdF9jaGFydF9sZWZ0IC5zZW5zb3JfY29tcGFyZSBtZC1zZWxlY3Qge1xuICAgIHRyYW5zZm9ybTogdHJhbnNsYXRlWSg4cHgpO1xuICAgIG1hcmdpbjogMDtcbiAgfVxuICBib2R5IC5raXRfZGV0YWlsZWQgLmtpdF9jaGFydCAuY29udGFpbmVyIC5raXRfY2hhcnRfbGVmdCAuc2Vuc29yX2RhdGEge1xuICAgIG1hcmdpbjogNSUgMCA0JSAwO1xuICB9XG5cbi8qICAgc2VjdGlvbi5tYXAge1xuICAgIHotaW5kZXg6IDA7XG4gIH0gKi9cbn1cblxuLyoqXG4gKiAgSU5QVVQgQ1NTXG4gKi9cbi8vXG4vLyA6Oi13ZWJraXQtaW5wdXQtcGxhY2Vob2xkZXIgeyAvKiBXZWJLaXQgYnJvd3NlcnMgKi9cbi8vICAgY29sb3I6ICRmb250X2NvbG9yX3doaXRlO1xuLy8gfVxuLy8gOi1tb3otcGxhY2Vob2xkZXIgeyAvKiBNb3ppbGxhIEZpcmVmb3ggNCB0byAxOCAqL1xuLy8gICBjb2xvcjogJGZvbnRfY29sb3Jfd2hpdGU7XG4vLyAgIG9wYWNpdHk6ICAxO1xuLy8gfVxuLy8gOjotbW96LXBsYWNlaG9sZGVyIHsgLyogTW96aWxsYSBGaXJlZm94IDE5KyAqL1xuLy8gICBjb2xvcjogICRmb250X2NvbG9yX3doaXRlO1xuLy8gICBvcGFjaXR5OiAgMTtcbi8vIH1cbi8vIDotbXMtaW5wdXQtcGxhY2Vob2xkZXIgeyAvKiBJbnRlcm5ldCBFeHBsb3JlciAxMCsgKi9cbi8vICAgY29sb3I6ICRmb250X2NvbG9yX3doaXRlO1xuLy8gfVxuLy9cbi8vIDpmb2N1czo6LXdlYmtpdC1pbnB1dC1wbGFjZWhvbGRlciB7IC8qIFdlYktpdCBicm93c2VycyAqL1xuLy8gICBjb2xvcjogJGJhY2tncm91bmRfYmx1ZV9kYXJrX2JhY2tncm91bmQ7XG4vLyB9XG4vLyA6Zm9jdXM6LW1vei1wbGFjZWhvbGRlciB7IC8qIE1vemlsbGEgRmlyZWZveCA0IHRvIDE4ICovXG4vLyAgIGNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2RhcmtfYmFja2dyb3VuZDtcbi8vICAgb3BhY2l0eTogIDE7XG4vLyB9XG4vLyA6Zm9jdXM6Oi1tb3otcGxhY2Vob2xkZXIgeyAvKiBNb3ppbGxhIEZpcmVmb3ggMTkrICovXG4vLyAgIGNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2RhcmtfYmFja2dyb3VuZDtcbi8vICAgb3BhY2l0eTogIDE7XG4vLyB9XG4vLyA6Zm9jdXM6LW1zLWlucHV0LXBsYWNlaG9sZGVyIHsgLyogSW50ZXJuZXQgRXhwbG9yZXIgMTArICovXG4vLyAgIGNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2RhcmtfYmFja2dyb3VuZDtcbi8vIH1cblxuaHRtbCB7XG4gIGZvbnQtZmFtaWx5OiAnUm9ib3RvIENvbmRlbnNlZCc7XG4gIGZvbnQtc2l6ZTogMThweDtcbn1cblxuYm9keSB7XG4gIGJhY2tncm91bmQtY29sb3I6IHJnYmEoMCwgMCwgMCwgMCk7XG4gIGNvbG9yOiAkb2ZmX2JsYWNrO1xufVxuXG5hLmFib3V0e1xuICBjdXJzb3I6IHBvaW50ZXI7XG4gIHRleHQtZGVjb3JhdGlvbjogbm9uZTtcbiAgY29sb3I6ICRibHVlO1xufVxuXG5saS5wb2xpY3ktdG9jIHtcbiAgcGFkZGluZzogM3B4O1xuICBhIHtcbiAgICB0ZXh0LWRlY29yYXRpb246IG5vbmU7XG4gIH1cbn1cblxudWwjcG9saWN5LXRvYyB7XG4gIHBhZGRpbmctYm90dG9tOiAwcHggIWltcG9ydGFudDtcbiAgcGFkZGluZy1ib3R0b206IDBweCAhaW1wb3J0YW50O1xufVxuXG4ucG9saWN5LXRvYyNoZWFkZXJ7XG4gIGxpc3Qtc3R5bGU6IG5vbmU7XG59XG5cbi5pY29uX2xhYmVse1xuICB3aWR0aDogMTRweDtcbiAgaGVpZ2h0OiAxNHB4O1xuICBtYXJnaW4tcmlnaHQ6IDRweDtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xuICB0b3A6IC0ycHg7XG4gIHN2ZyB7XG4gICAgZmlsbDogJGdyZXlfZGFya2VyO1xuICB9XG59XG5cbi8qKlxuICAqIFRZUE9HUkFQSFlcbiAgKi9cblxuaDF7XG4gIGZvbnQtc2l6ZTogMi4yZW07XG4gIGZvbnQtd2VpZ2h0OiBub3JtYWw7XG59XG5oMiB7XG4gIGZvbnQtc2l6ZTogMS43ZW07XG4gIGZvbnQtd2VpZ2h0OiBub3JtYWw7XG59XG5oMyB7XG4gIGZvbnQtc2l6ZTogMS40ZW07XG4gIGZvbnQtd2VpZ2h0OiBub3JtYWw7XG59XG5oNCB7XG4gIGZvbnQtc2l6ZTogMS4yZW07XG4gIGZvbnQtd2VpZ2h0OiBub3JtYWw7XG59XG5cblxucCB7XG4gIGxpbmUtaGVpZ2h0OiAxLjU7XG59XG5cbnRkIHtcbiAgcGFkZGluZzo1cHg7XG59XG5cbmlucHV0e1xuICBmb250LWZhbWlseTonUm9ib3RvIENvbmRlbnNlZCc7XG59XG5cbi5sYWJlbCwgLnRhZywgLnN0YXRle1xuICB0ZXh0LXRyYW5zZm9ybTogdXBwZXJjYXNlO1xuICBmb250LXNpemU6IDEycHg7XG4gIGxpbmUtaGVpZ2h0OjI0cHg7XG4gIGJvcmRlci1yYWRpdXM6MjBweDtcbiAgcGFkZGluZzogM3B4IDhweDtcbiAgbWFyZ2luLXJpZ2h0OiA4cHg7XG5cbiAgbWQtaWNvbntcbiAgICBtYXJnaW4tbGVmdDo0cHg7XG4gICAgaGVpZ2h0OjExcHg7XG4gICAgd2lkdGg6MTFweDtcbiAgICBtYXJnaW4tdG9wOi0zcHg7XG4gICAgY3Vyc29yOnBvaW50ZXI7XG4gIH1cbiAgJltjbGlja2FibGVde1xuICAgIGN1cnNvcjpwb2ludGVyO1xuICB9XG59XG4ubGFiZWx7XG4gIGJhY2tncm91bmQtY29sb3I6ICR5ZWxsb3dfbGlnaHQ7XG4gIGNvbG9yOiAkb2ZmX2JsYWNrO1xuICBtZC1pY29uIHN2ZyBwYXRoe1xuICAgIGZpbGw6ICR0ZXJjaWFyeV9jb2xvcl9saWdodDtcbiAgfVxufVxuLnRhZyB7XG4gIGJhY2tncm91bmQtY29sb3I6ICR5ZWxsb3dfbGlnaHRlcjtcbiAgY29sb3I6ICRvZmZfYmxhY2s7XG4gIG1kLWljb24gc3ZnIHBhdGh7XG4gICAgZmlsbDogJHNlY29uZGFyeV9jb2xvcjtcbiAgfVxufVxuXG4ubGVhZmxldC1jb250YWluZXIgLnRhZywgLmxlYWZsZXQtY29udGFpbmVyIC5sYWJlbCxcbi5sZWFmbGV0LWNvbnRhaW5lciAuc3RhdGV7XG4gIGZvbnQtc2l6ZToxZW07XG59XG5cbi5sZWFmbGV0LXRvcHtcbiAgIHRvcDo5M3B4O1xufVxuXG5zZWN0aW9uLmluZm97XG4gIGgxe1xuICAgIGZvbnQtc2l6ZToxLjNlbTtcbiAgfVxufVxuXG5he1xuICBjb2xvcjogJG9mZl9ibGFjaztcbiAgJjpob3ZlciwgJjphY3RpdmV7XG4gICAgdGV4dC1kZWNvcmF0aW9uOiBub25lO1xuICAgIGNvbG9yOiAkYmx1ZTtcbiAgfVxufVxuXG5hLmZvb3Rlci1saW5re1xuICBjb2xvcjogJHllbGxvdztcbiAgdGV4dC1kZWNvcmF0aW9uOiBub25lO1xuICB0cmFuc2l0aW9uOiBjb2xvciAwLjRzIGN1YmljLWJlemllcigwLjI1LCAwLjgsIDAuMjUsIDEpO1xuICAmOmhvdmVyLCAmOmFjdGl2ZXtcbiAgICB0ZXh0LWRlY29yYXRpb246IG5vbmU7XG4gICAgY29sb3I6JGJsdWU7XG4gIH1cbn1cbi8vIGEubWQtYnV0dG9uLCBhLFxuLy8gYnV0dG9uLm1kLWJ1dHRvbiwgYnV0dG9ue1xuLy8gICAvLyBjb2xvciBoZXJlIG92ZXJ3cml0ZXMgTUFQIGFuZCBDT01NVU5JVFkgaW4gbmF2YmFyXG4vLyAgIC8vIGNvbG9yOiAkdGVyY2lhcnlfY29sb3I7XG4vLyB9XG5cbi8qXG4gIE92ZXJ3cml0ZSBnbG9iYWwgbWF0ZXJpYWwgZGVzaWduLCBhbGxvdyBpY29ucyB0byBiZSBzbWFsbGVyXG4qL1xuXG5tZC10b29sYmFye1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2RhcmtfbmF2YmFyICFpbXBvcnRhbnQ7XG4gIHotaW5kZXg6ICRsYXJnZV9wcmlvcml0eTtcbiAgY29sb3I6IHdoaXRlICFpbXBvcnRhbnQ7XG4gIG1kLWljb257XG4gICAgZmlsbDogd2hpdGUgIWltcG9ydGFudDtcbiAgfVxuICAvL2hlaWdodDogNjRweDtcbn1cblxubWQtaWNvbntcbiAgLy8gRWZmZWN0cyBBTEwgaWNvbnMgb24gdGhlIHNpdGVcbiAgbWluLXdpZHRoOiB1bnNldCAhaW1wb3J0YW50O1xufVxuXG5tZC1zZWxlY3R7XG4gIG1kLXNlbGVjdC12YWx1ZXtcbiAgICBib3JkZXItYm90dG9tLWNvbG9yOiAjY2NjY2NjNzUgIWltcG9ydGFudDtcbiAgfVxufVxuXG5hLm1kLWJ1dHRvbiBoNCwgYS5tZC1idXR0b24gLmg0e1xuICBjb2xvcjogJHRlcmNpYXJ5X2NvbG9yO1xuICBtYXJnaW46IDA7XG59XG5hLndhcm4sXG5idXR0b24ud2FybntcbiAgY29sb3I6ICRhbGVydF9yZWQ7XG4gIC8vIHN2ZyB7XG4gIC8vICAgLy8gZmlsbDogJGFsZXJ0X3JlZDtcbiAgLy8gfVxufVxuXG4uYm9sZHtcbiAgZm9udC13ZWlnaHQ6IGJvbGQ7XG59XG5cbmRpdlthcGkta2V5XXtcbiAgZGlzcGxheTogZmxleDtcblxuICAuYXBpX2tleV9udW1iZXIge1xuICAgIGJhY2tncm91bmQtY29sb3I6ICNGMkYyRjI7XG4gICAgd29yZC1icmVhazogYnJlYWstYWxsO1xuICAgIHdpZHRoOiA0MjBweDtcbiAgfVxuICAuYXBpX2tleV9yZWZyZXNoX2J1dHRvbiB7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogI0RCREJEQjtcbiAgICBtYXJnaW46MDtcbiAgICBtZC1pY29uIHtcbiAgICAgIHBhZGRpbmc6IDEwcHg7XG4gICAgICBoZWlnaHQ6IDE1cHg7XG4gICAgICB3aWR0aDogMTVweDtcbiAgICB9XG4gIH1cbiAgbWQtaWNvbiB7XG4gICAgb3BhY2l0eTogLjU7XG4gIH1cbn1cblxuLmluZm9fb3ZlcmxheXtcbiAgaDJ7XG4gICAgY29sb3I6d2hpdGU7XG4gIH1cbiAgcHtcbiAgICBtYXJnaW46IDA7XG4gICAgZm9udC1zaXplOiAyMHB4O1xuICAgIGZvbnQtd2VpZ2h0OiAzMDA7XG4gICAgY29sb3I6JHRlcmNpYXJ5X2NvbG9yX2xpZ2h0O1xuICB9XG59XG5cbi5kYXJre1xuICBoMSwgaDJ7XG4gICAgY29sb3I6d2hpdGU7XG4gIH1cbiAgaDEsIGgye1xuICAgIG1hcmdpbjowO1xuICB9XG4gIGgzLCBoNCwgLmg0LCBhe1xuICAgIGNvbG9yOiAkZm9udF9jb2xvcl9ibHVlX2dyZXk7XG4gIH1cbn1cblxuLyoqXG4gKiBBTkdVTEFSIE1BVEVSSUFMIE9WRVJSSURJTkdcbiAqL1xuXG5cbm1kLXNlbGVjdC1tZW51IHtcbiAgYmFja2dyb3VuZC1jb2xvcjogd2hpdGU7XG59XG5tZC1zZWxlY3QsXG5tZC1zZWxlY3QtbWVudSB7XG4gIHRleHQtdHJhbnNmb3JtOiB1cHBlcmNhc2U7XG59XG5tZC1zZWxlY3Q6bm90KFtkaXNhYmxlZF0pOmZvY3VzIC5tZC1zZWxlY3QtdmFsdWUge1xuICBib3JkZXItYm90dG9tLWNvbG9yOiAjNkI4NjhEO1xufVxuXG4vLyBidXR0b24ubWQtcHJpbWFyeXtcbi8vICAgLy8gR2V0IG5ldyBraXQgcG9wdXAsIGJ1dHRvbiBjb2xvclxuLy8gICAvLyBBbHNvIGludGVyZmVyZXMgd2l0aCBraXRzL2VkaXQgc2F2ZSBidXR0b25cbi8vICAgLy8gY29sb3I6JHRlcmNpYXJ5X2NvbG9yICFpbXBvcnRhbnQ7XG4vLyAgIC8vIGZvbnQtc2l6ZToxOHB4O1xuLy8gfVxuLy8gLm1kLWJ1dHRvbi5tZC1mbGF0Lm1kLXByaW1hcnkge1xuLy8gICAvL21hcmdpbjogMHB4O1xuLy8gICAvL2JhY2tncm91bmQtY29sb3I6ICR0ZXJjaWFyeV9jb2xvcjtcbi8vICAgLy9jb2xvcjogd2hpdGUgIWltcG9ydGFudDtcbi8vIH1cblxuLm1kLWJ1dHRvbi5tZC1wcmltYXJ5Lm1kLWZsYXQ6bm90KFtkaXNhYmxlZF0pOmhvdmVyIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJHRlcmNpYXJ5X2NvbG9yO1xuICBjb2xvcjogd2hpdGU7XG59XG5cbi5tZC1idXR0b24ubWQtcHJpbWFyeS5tZC1mbGF0Om5vdChbZGlzYWJsZWRdKTpmb2N1cyB7XG4gIGJhY2tncm91bmQtY29sb3I6ICR0ZXJjaWFyeV9jb2xvcjtcbiAgY29sb3I6IHdoaXRlO1xufVxuXG4ubWQtYnV0dG9uLmZpbGx3aWR0aCB7XG4gIHdpZHRoOiAxMDAlICFpbXBvcnRhbnQ7XG59XG5cbm1kLWlucHV0LWNvbnRhaW5lcjpub3QoLm1kLWlucHV0LWludmFsaWQpLm1kLWlucHV0LWZvY3VzZWQge1xuICBsYWJlbCB7XG4gICAgY29sb3I6ICR0ZXJjaWFyeV9jb2xvcjtcbiAgfVxuXG4gIC5tZC1pbnB1dCB7XG4gICAgYm9yZGVyLWNvbG9yOiByZ2JhKDAsMCwwLDAuMTIpO1xuICB9XG59XG5cbi8qKlxuICogQUxFUlQgQ09NUE9ORU5UXG4gKi9cblxuLm1kLXRvYXN0LWNvbnRlbnQge1xuICBib3gtc2hhZG93OiB1bnNldCAhaW1wb3J0YW50O1xuICBiYWNrZ3JvdW5kLWNvbG9yOiB1bnNldCAhaW1wb3J0YW50O1xuICBjb2xvcjogYmxhY2sgIWltcG9ydGFudDtcbiAgbWluLXdpZHRoOiAxMDAlO1xufVxuXG5tZC10b2FzdCB7XG4gIC8vIFRPRE86IEJ1ZyB3aGVuIGEgdG9hc3QgaXMgb24gc2NyZWVuLCB0aGUgYm9keSBnZXRzIGNsYXNzPSdtZC10b2FzdC1vcGVuLWJvdHRvbSdcbiAgLy8gV2hlcmUgZG9lcyB0aGF0IGNvbWUgZnJvbT9cbiAgLy8gSWYgdGhlcmUgaXMgYSBzaW1pbGFyICpvcGVuLXRvcCwgd2UgY2FuIHJlbW92ZSB0aGUgQ1NTIGJvdHRvbSwgdG9wXG4gIGJvdHRvbTogdW5zZXQgIWltcG9ydGFudDsgLy8gT3ZlcndyaXRlIGFuZ3VsYXIgZGVmYXVsdFxuICBjb2xvcjogYmxhY2s7XG4gIG9wYWNpdHk6IDAuOTU7XG4gIHBhZGRpbmc6IDA7XG4gIHBvc2l0aW9uOiBmaXhlZDtcbiAgdG9wOiA2NHB4ICFpbXBvcnRhbnQ7IC8vIFN0YXJ0IHRoZSB0b2FzdCBhdCB0b3AgYmVsb3cgbmF2YmFyLlxuICBtaW4td2lkdGg6IDEwMCU7XG4gIHotaW5kZXg6IDE1O1xuXG4gIC5hbGVydF90eXBlSWNvbiB7XG4gICAgd2lkdGg6IDE2cHg7XG4gICAgaGVpZ2h0OiAxNnB4O1xuICAgIG1hcmdpbi1yaWdodDogMTZweDtcbiAgfVxuXG4gIC5hbGVydF9jbG9zZUljb24ge1xuICAgIHdpZHRoOiAxMC41cHg7XG4gICAgaGVpZ2h0OiAxMC41cHg7XG4gICAgbWFyZ2luOiAwIGF1dG87XG4gIH1cblxuICAuYWxlcnRfYnV0dG9uIHtcbiAgICBwYWRkaW5nOiAwcHggMTJweDtcbiAgICBtYXJnaW4tbGVmdDogNXB4O1xuICAgIGNvbG9yOiBibGFjayAhaW1wb3J0YW50O1xuICAgIGJhY2tncm91bmQtY29sb3I6ICR5ZWxsb3c7XG4gICAgYm9yZGVyLXJhZGl1czogMjBweDtcbiAgICBmb250LXNpemU6IDEycHg7XG4gIH1cbn1cblxubWQtdG9hc3QucmVkIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJGFsZXJ0X3JlZDtcbn1cblxubWQtdG9hc3QuZ3JlZW4ge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkYWxlcnRfZ3JlZW47XG59XG5cbm1kLXRvYXN0LmJsdWUge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmx1ZTtcbn1cblxubWQtdG9hc3QueWVsbG93IHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJHllbGxvdztcbn1cblxubWQtY2hlY2tib3gubWQtY2hlY2tlZCAubWQtaWNvbiB7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogJGJsdWU7XG59XG5tZC1jaGVja2JveCAubWQtbGFiZWwge1xuICAgIHBvaW50ZXItZXZlbnRzOiBhdXRvO1xufVxubWQtY2hlY2tib3ggLm1kLWxhYmVsIHNwYW4ge1xuICAgIHBvaW50ZXItZXZlbnRzOiBub25lO1xuICAgIC13ZWJraXQtdXNlci1zZWxlY3Q6IG5vbmU7XG4gICAgLW1vei11c2VyLXNlbGVjdDogbm9uZTtcbiAgICAtbXMtdXNlci1zZWxlY3Q6IG5vbmU7XG4gICAgdXNlci1zZWxlY3Q6IG5vbmU7XG59XG5cbi8qKlxuICogIFNQSU5ORVIvUFJPR1JFU1MgQkFSIENPTVBPTkVOVFNcbiAqL1xuXG5tZC1wcm9ncmVzcy1saW5lYXIubWQtYmFyLmdyZWVuIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJGdyZWVuO1xufVxubWQtcHJvZ3Jlc3MtbGluZWFyLmdyZWVuIC5tZC1jb250YWluZXIge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiByZ2JhKCRncmVlbiwgMCk7XG59XG5tZC1wcm9ncmVzcy1saW5lYXIuZ3JlZW4gLm1kLWJhciB7XG4gIGJhY2tncm91bmQtY29sb3I6ICRncmVlbjtcbn1cbm1kLXByb2dyZXNzLWxpbmVhci5tZC1iYXJ7XG4gIGJhY2tncm91bmQtY29sb3I6ICRibHVlO1xufVxubWQtcHJvZ3Jlc3MtbGluZWFyLm1kLWh1ZS0zIC5tZC1jb250YWluZXJ7XG4gIGJhY2tncm91bmQtY29sb3I6IHJnYmEoJGJsdWUsIDAuMDUpO1xufVxubWQtcHJvZ3Jlc3MtbGluZWFyLm1kLWh1ZS0zIC5tZC1iYXIge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmx1ZTtcbn1cblxuLmtpdF9zcGlubmVyIHtcbiAgei1pbmRleDogMjAwO1xuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIGxlZnQ6IDA7XG4gIGJvdHRvbTogNnB4O1xufVxuXG4uY2hhcnRfc3Bpbm5lciB7XG4gIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgdG9wOiAyMDBweDtcbiAgbGVmdDogNTclO1xufVxuXG4vKipcbiAqICBDT09LSUVTIFBPTElDWSBDT01QT05FTlRcbiAqL1xuXG4gLmNvb2tpZXMtcG9saWN5X2NvbnRhaW5lcntcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiByZ2JhKDYwLCA2MCwgNjAsIDEpO1xuICAgIGNvbG9yOiAkeWVsbG93O1xuICAgIHBvc2l0aW9uOiBmaXhlZDtcbiAgICBib3R0b206IDA7XG4gICAgcGFkZGluZzo1cHg7XG4gICAgbGVmdDogMDtcbiAgICB3aWR0aDogMTAwJTtcbiAgICAvL2hlaWdodDogNHZoO1xuICAgIHotaW5kZXg6IDk5OTtcbiAgICB0ZXh0LWFsaWduOiBjZW50ZXI7XG4gICAgbGluZS1oZWlnaHQ6IDR2aDtcblxuICAgIGF7XG4gICAgICBjb2xvcjogcmdiYSgkeWVsbG93LCAwLjYpO1xuICAgIH1cbiB9XG5cblxuLy8gTm8gZGF0YSBiYWNrZHJvcFxuLm1kLW5vRGF0YUJhY2tkcm9wIHtcbiAgd2lkdGg6IDEwMCU7XG4gIGhlaWdodDogOTYgKyAyODVweDtcbiAgcG9zaXRpb246IGFic29sdXRlO1xuICBiYWNrZ3JvdW5kLWNvbG9yOiB3aGl0ZTtcblxuICAuYmxvY2sge1xuICAgIG1hcmdpbjogMCBhdXRvO1xuXG4gIH1cbiAgLnRpdGxlLCAubWVzc2FnZSB7XG4gICAgdGV4dC1hbGlnbjogY2VudGVyO1xuICB9XG4gIC50aXRsZSB7XG4gICAgY29sb3I6ICRncmV5X2RhcmtlcjtcbiAgICBmb250LXdlaWdodDogNDAwO1xuICAgIG1hcmdpbi1ib3R0b206IDA7XG4gIH1cbiAgLm1lc3NhZ2Uge1xuICAgIG1hcmdpbjogMDtcbiAgICBmb250LXNpemU6IDIwcHg7XG4gICAgZm9udC13ZWlnaHQ6IDMwMDtcbiAgfVxuICBhIHtcbiAgICBjb2xvcjogJGJsdWU7XG4gICAgdGV4dC1kZWNvcmF0aW9uOiBub25lO1xuICB9XG59XG5cbi8qKlxuICogIE1BUFxuICovXG5cbi5tYXBfc3RhdGUge1xuICBwb3NpdGlvbjogZml4ZWQ7XG4gIHotaW5kZXg6IDE7XG59XG5cbi8vIEBtZWRpYSBvbmx5IHNjcmVlbiBhbmQgKG1heC1kZXZpY2Utd2lkdGg6IDEwMjRweCkge1xuLy8gICAubWFwX3N0YXRlIHtcbi8vICAgfVxuLy8gfVxuXG5zZWN0aW9uLm1hcCB7XG4gIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xuICB0ZXh0LWFsaWduOiBjZW50ZXI7XG4gIHdpZHRoOiAxMDAlO1xuICBoZWlnaHQ6IDEwMCU7XG4gIGNvbG9yOiBibGFjaztcbiAgei1pbmRleDogMTsvLyRzbWFsbF9wcmlvcml0eTtcbiAgcG9zaXRpb246IGZpeGVkO1xuICB0b3A6IDY0cHg7XG5cbiAgQG1lZGlhIGFsbCBhbmQgKG1heC1oZWlnaHQ6IDEyMDBweCkgYW5kIChtaW4taGVpZ2h0OiAxMDAwcHgpIHtcbiAgICBoZWlnaHQ6IDk1MHB4O1xuICB9XG4gIEBtZWRpYSBhbGwgYW5kIChtYXgtaGVpZ2h0OiAxMDAwcHgpIGFuZCAobWluLWhlaWdodDogODAwcHgpIHtcbiAgICBoZWlnaHQ6IDcwMHB4O1xuICB9XG4gIEBtZWRpYSBhbGwgYW5kIChtYXgtaGVpZ2h0OiA4MDBweCkgYW5kIChtaW4taGVpZ2h0OiA2MDBweCkge1xuICAgIGhlaWdodDogNTUwcHg7XG4gIH1cbiAgQG1lZGlhIGFsbCBhbmQgKG1heC1oZWlnaHQ6IDYwMHB4KSBhbmQgKG1pbi1oZWlnaHQ6IDQwMHB4KSB7XG4gICAgaGVpZ2h0OiAzMDBweDtcbiAgfVxuXG5cbiAgaDEge1xuICAgIGZvbnQtc2l6ZTogM2VtO1xuICB9XG59XG5cbi8qXG4gIENPUkUgQ0hJUCBDT01QT05FTlRcbiAqL1xuLmNoaXBzIHtcbiAgZGlzcGxheTogaW5saW5lLWJsb2NrO1xuXG4gIGRpdi5jaGlwc19yb3d7XG4gICAgaGVpZ2h0OjQwLjVweDtcbiAgfVxuXG4gIC5sYWJlbCwgLnRhZ3tcbiAgICBib3gtc2hhZG93OiAwIDFweCA1cHggcmdiYSgwLDAsMCwwLjY1KTtcbiAgfVxufVxuLmNoaXAge1xuICAvLyBkaXNwbGF5OiBpbmxpbmUtYmxvY2s7XG4gIHBhZGRpbmc6IDBweCAxMHB4IDBweCAxNHB4O1xuICBtYXJnaW46IGF1dG8gNXB4O1xuICBib3JkZXItcmFkaXVzOiAyMHB4O1xuICBoZWlnaHQ6MzBweDtcbiAgdGV4dC10cmFuc2Zvcm06IHVwcGVyY2FzZTtcbiAgZm9udC1zaXplOiAxM3B4O1xuICBsaW5lLWhlaWdodDogMjlweDtcblxuICAuY2hpcF9uYW1lIHtcbiAgICBtYXJnaW4tbGVmdDogM3B4O1xuICAgIG1hcmdpbi1yaWdodDogN3B4O1xuICB9XG4gIC5jaGlwX2ljb24ge1xuICAgIGJvcmRlcjogMDtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiBpbmhlcml0O1xuICAgIG1kLWljb24ge1xuICAgICAgd2lkdGg6IDhweDtcbiAgICAgIGhlaWdodDogOHB4O1xuICAgIH1cbiAgfVxuXG4gICYubWFwX2ZpbHRlciB7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogd2hpdGU7XG4gICAgYm94LXNoYWRvdzogMCAxcHggNXB4IHJnYmEoMCwwLDAsMC42NSk7XG4gIH1cbiAgJi5raXRfdGFnIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkZ3JleTtcbiAgICBmbG9hdDogbGVmdDtcbiAgICBtYXJnaW46IDVweDtcbiAgfVxufVxuLypcbiAgTUFQIEZJTFRFUlNcbiovXG4ubWFwX2xlZ2VuZCB7XG5cbiAgcG9zaXRpb246IGFic29sdXRlO1xuICB0b3A6IDEwcHg7XG4gIG1hcmdpbi1sZWZ0OiAxMHB4O1xuICBkaXNwbGF5OiBmbGV4O1xuXG4gIC5jaGlwc3tcbiAgICBhbGlnbi1zZWxmOiBmbGV4LWVuZDtcbiAgICBwYWRkaW5nLWJvdHRvbTogNXB4O1xuICAgIHBhZGRpbmctbGVmdDogNXB4O1xuICAgIGRpc3BsYXk6IGZsZXg7XG4gIH1cblxuICAubWFwX2xlZ2VuZF9fZmlsdGVyc0NvbnRhaW5lcntcbiAgICBkaXNwbGF5OiBmbGV4O1xuICAgIGZsZXgtZGlyZWN0aW9uOiBjb2x1bW47XG4gICAganVzdGlmeS1jb250ZW50OiBzcGFjZS1iZXR3ZWVuO1xuICAgIHdpZHRoOiAxMjRweDtcbiAgICBoZWlnaHQ6IDgxcHg7XG5cbiAgICBib3JkZXItcmFkaXVzOjRweDtcbiAgICBvdmVyZmxvdzpoaWRkZW47XG4gICAgYm94LXNoYWRvdzogMCAxcHggNXB4IHJnYmEoMCwgMCwgMCwgMC42NSk7XG4gICAgc3ZnLCBwIHtcbiAgICAgIGZpbGw6ICRvZmZfYmxhY2s7XG4gICAgfVxuXG5cbiAgICAubWFwX2xlZ2VuZF9fZmlsdGVyc1Jvd3tcbiAgICAgIGN1cnNvcjogcG9pbnRlcjtcblxuICAgICAgJjpub3QoOmxhc3QtY2hpbGQpe1xuICAgICAgICBib3JkZXItYm90dG9tOiAxcHggc29saWQgZ3JheTtcbiAgICAgIH1cblxuICAgICAgZGlzcGxheTogZmxleDtcblxuICAgICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgICAgIHBhZGRpbmctbGVmdDogNXB4O1xuICAgICAgYmFja2dyb3VuZC1jb2xvcjogI2ZmZjtcblxuICAgICAgLm1hcF9maWx0ZXJfYnV0dG9ue1xuICAgICAgICBib3gtc2hhZG93Om5vbmU7XG4gICAgICAgIHB7XG4gICAgICAgICAgbWFyZ2luOiAwO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuICB9XG59XG4ubWFwX2ZpbHRlcl9idXR0b24ge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiB3aGl0ZTtcbiAgbWFyZ2luLXJpZ2h0OiAxNXB4O1xuICB3aWR0aDogMjZweDtcbiAgaGVpZ2h0OiAyNnB4O1xuICBib3gtc2hhZG93OiAwIDFweCA1cHggcmdiYSgwLDAsMCwwLjY1KTtcblxuICBtZC1pY29uIHtcbiAgICB3aWR0aDogMThweDtcbiAgICBoZWlnaHQ6IDEycHg7XG4gIH1cbn1cbi5tZC1idXR0b24ubWFwX2ZpbHRlcl9idXR0b246aG92ZXIsIC5tZC1idXR0b24ubWFwX2ZpbHRlcl9idXR0b246Zm9jdXMge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiB3aGl0ZTtcbn1cblxuXG5cbi8qXG49PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuS0lUIFRBR1MgTU9EVUxFXG49PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuKi9cblxuLmtpdFRhZ3NfX3NlY3Rpb257XG4gIG1hcmdpbi10b3A6IDM3NnB4O1xuXG4gIC5raXRUYWdzX19saXN0Q29udGFpbmVye1xuICAgIGJhY2tncm91bmQtY29sb3I6ICRncmV5X2xpZ2h0ZXI7XG4gICAgcGFkZGluZzogMjBweDtcbiAgfVxuXG4gIC5raXRUYWdzX19jb250YWluZXJ7XG4gICAgcGFkZGluZy1ib3R0b206IDIwcHg7XG5cbiAgICAua2l0VGFnc19fdGV4dENvbnRhaW5lcntcbiAgICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgICBqdXN0aWZ5LWNvbnRlbnQ6IHNwYWNlLWJldHdlZW47XG4gICAgICBtYXJnaW4tbGVmdDogMjBweDtcbiAgICAgIG1hcmdpbi1yaWdodDogMjBweDtcbiAgICAgIC5raXRUYWdzX190ZXh0RWxlbWVudHtcbiAgICAgICAgZmxleC1iYXNpczo0MCU7XG4gICAgICAgIGZsZXgtZ3JvdzowO1xuICAgICAgICBtaW4td2lkdGg6IDMwMHB4O1xuICAgICAgfVxuICAgIH1cbiAgfVxufVxuXG4vKlxuIENPUkUgRk9STSBDT01QT05FTlRcbiAqL1xuXG4uZm9ybV9jb250YWluZXIge1xuICBmb250LXNpemU6IDE2cHg7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgYm9yZGVyLXJhZGl1czogMHB4IDBweCAzcHggM3B4O1xufVxuLmZvcm1fY29udGVudENvbnRhaW5lciB7XG4gIHdpZHRoOiA5MiU7XG4gIG1hcmdpbjogMCBhdXRvO1xufVxuLmZvcm1fZmllbGQge1xuICBwb3NpdGlvbjogcmVsYXRpdmU7XG59XG5cbi5mb3JtX2Vycm9ycyB7XG4gIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgYm90dG9tOiAxNSU7XG4gIGxlZnQ6IDUlO1xuICBjb2xvcjogcmVkO1xuICBmb250LXNpemU6IDAuN2VtO1xuICBsaW5lLWhlaWdodDogMTtcbn1cblxuLmZvcm1fY2xvc2VJY29uIHtcbiAgZmxvYXQ6IHJpZ2h0O1xuICB3aWR0aDogMTRweDtcbiAgaGVpZ2h0OiAxNHB4O1xuICBtYXJnaW46IDEycHg7XG5cbiAgbWQtaWNvbiB7XG4gICAgd2lkdGg6IDE0cHg7XG4gICAgaGVpZ2h0OiAxNHB4O1xuICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICB0b3A6IDBweDtcbiAgICBsZWZ0OiAwcHg7XG4gIH1cbn1cblxuLm1kLXByaW1hcnkuZm9ybV9idXR0b24ge1xuICBjb2xvcjogJGZvbnRfY29sb3Jfd2hpdGU7XG4gIGJhY2tncm91bmQtY29sb3I6ICR0ZXJjaWFyeV9jb2xvcjtcbiAgaGVpZ2h0OiA2NHB4O1xuICBib3R0b206IDBweDtcbiAgbGVmdDogMHB4O1xuICBib3JkZXItcmFkaXVzOiAwcHggMHB4IDJweCAycHg7XG59XG5cblxuLmZvcm1fdGl0bGUge1xuICBjb2xvcjogJHRlcmNpYXJ5X2NvbG9yO1xuICBmb250LXdlaWdodDogbm9ybWFsO1xufVxuXG4uZm9ybV9tZXNzYWdlSGVhZGVyIHtcbiAgZm9udC1zaXplOiAxLjVlbTtcbiAgY29sb3I6ICRzZWNvbmRhcnlfY29sb3I7XG4gIGZvbnQtd2VpZ2h0OiBib2xkO1xuICBsaW5lLWhlaWdodDogMDtcbiAgbWFyZ2luLXRvcDogNTBweDtcbn1cbi5mb3JtX21lc3NhZ2VTdWJoZWFkZXIge1xuICBmb250LXNpemU6IDAuODc1ZW07XG4gIGNvbG9yOiByZ2JhKDAsOTgsMTIzLDAuNTQpO1xuICBsaW5lLWhlaWdodDogMC41O1xufVxuXG4uZm9ybV9tZXNzYWdlRGVzY3JpcHRpb24ge1xuICBmb250LXNpemU6IDEycHg7XG4gIG1hcmdpbi1ib3R0b206IDA7XG59XG5cbi8qXG4gIENPUkUgTEFSR0UgREVWSUNFUyBGT1JNXG4qL1xuQG1lZGlhIChtaW4td2lkdGg6IDUwMXB4KSB7XG4gIC5mb3JtX2NvbnRhaW5lciB7XG4gICAgd2lkdGg6IDM0MHB4O1xuICAgIGZvbnQtc2l6ZTogMTZweDtcbiAgfVxuICAuZm9ybV9tZXNzYWdlQ29udGFpbmVyIHtcbiAgICBtYXJnaW4tYm90dG9tOiAyMHB4O1xuICB9XG4gIC5tZC1wcmltYXJ5LmZvcm1fYnV0dG9uIHtcbiAgICB3aWR0aDogMzQwcHg7XG4gICAgZGlzcGxheTogYmxvY2s7XG4gICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgIG1hcmdpbjogMCBhdXRvO1xuICAgIGZvbnQtc2l6ZTogMS41ZW07XG4gIH1cbn1cblxuLypcbiAgQ09SRSBNT0JJTEUgREVWSUNFUyBGT1JNXG4qL1xuXG5AbWVkaWEgKG1heC13aWR0aDogNTAwcHgpIHtcbiAgLmZvcm1fY29udGFpbmVyIHtcbiAgICB3aWR0aDogODAlO1xuICAgIGZvbnQtc2l6ZTogMTJweDtcbiAgfVxuICAubWQtcHJpbWFyeS5mb3JtX2J1dHRvbiB7XG4gICAgd2lkdGg6IDEwMCU7XG4gICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgIGZvbnQtc2l6ZTogMS4yZW07XG4gIH1cbn1cblxuLy8gb3ZlcnJpZGUgZm9yIElwaG9uZSA0XG5AbWVkaWEgb25seSBzY3JlZW4gYW5kIChtaW4tZGV2aWNlLXdpZHRoOiAzMjBweCkgYW5kIChtYXgtZGV2aWNlLXdpZHRoOiA0ODBweCkgYW5kICgtd2Via2l0LWRldmljZS1waXhlbC1yYXRpbzogMikgYW5kIChkZXZpY2UtYXNwZWN0LXJhdGlvOiAyLzMpIHtcbiAgLnJlY292ZXJ5X2NvbnRhaW5lciB7XG4gICAgbWFyZ2luLXRvcDogNDBweDtcbiAgfVxuICAuZm9ybV9maWVsZCB7XG4gICAgcGFkZGluZzogNSUgMDtcbiAgfVxufVxuXG4vKlxuICBSRUNPVkVSWSBGT1JNXG4qL1xuLnJlY292ZXJ5X2NvbnRhaW5lciB7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgbWFyZ2luOiAwcHggYXV0bztcbiAgbWFyZ2luLWJvdHRvbToxMDBweDtcbi8vICBoZWlnaHQ6IDM3NnB4O1xufVxuXG4uZm9ybVJlY292ZXJ5X2ZpZWxkIHtcbiAgcGFkZGluZzogMTAlIDA7XG59XG5cbkBtZWRpYSAobWluLXdpZHRoOiA1MDFweCkge1xuICAucmVjb3ZlcnlfY29udGFpbmVyIHtcbiAgICBtYXJnaW4tdG9wOiA4MHB4O1xuICB9XG4gIC5mb3JtUmVjb3ZlcnlfZXJyb3JzIHtcbiAgICBib3R0b206IDE1JTtcbiAgICBsZWZ0OiA1JTtcbiAgfVxufVxuXG5AbWVkaWEgKG1heC13aWR0aDogNTAwcHgpIHtcbiAgLnJlY292ZXJ5X2NvbnRhaW5lciB7XG4gICAgbWFyZ2luLXRvcDogMjAlO1xuICB9XG4gIC5mb3JtUmVjb3ZlcnlfZXJyb3JzIHtcbiAgICBib3R0b206IDIzJTtcbiAgICBsZWZ0OiA1JTtcbiAgfVxufVxuXG5AbWVkaWEgb25seSBzY3JlZW4gYW5kIChtaW4tZGV2aWNlLXdpZHRoOiAzMjBweCkgYW5kIChtYXgtZGV2aWNlLXdpZHRoOiA0ODBweCkgYW5kICgtd2Via2l0LWRldmljZS1waXhlbC1yYXRpbzogMikgYW5kIChkZXZpY2UtYXNwZWN0LXJhdGlvOiAyLzMpIHtcblxuICBAbWVkaWEgKG1heC13aWR0aDogNTAwcHgpIHtcbiAgICAuZm9ybVJlY292ZXJ5X2ZpZWxkIHtcbiAgICAgIHBhZGRpbmc6IDIlIDA7XG4gICAgfVxuICAgIC5mb3JtUmVjb3ZlcnlfZXJyb3JzIHtcbiAgICAgIGJvdHRvbTogMjklO1xuICAgIH1cbiAgfVxufVxuXG4vKlxuICBSRVNFVCBGT1JNXG4qL1xuLmZvcm1SZXNldF9maWVsZCB7XG4gIHBhZGRpbmc6IDclIDA7XG59XG5cbkBtZWRpYSAobWluLXdpZHRoOiA1MDFweCkge1xuICAuZm9ybVJlc2V0X2Vycm9ycyB7XG4gICAgYm90dG9tOiA1JTtcbiAgICBsZWZ0OiA1JTtcbiAgfVxufVxuXG5AbWVkaWEgKG1heC13aWR0aDogNTAwcHgpIHtcbiAgLmZvcm1SZXNldF9lcnJvcnMge1xuICAgIGJvdHRvbTogMTglO1xuICAgIGxlZnQ6IDElO1xuICB9XG59XG5cbkBtZWRpYSBvbmx5IHNjcmVlbiBhbmQgKG1pbi1kZXZpY2Utd2lkdGg6IDMyMHB4KSBhbmQgKG1heC1kZXZpY2Utd2lkdGg6IDQ4MHB4KSBhbmQgKC13ZWJraXQtZGV2aWNlLXBpeGVsLXJhdGlvOiAyKSBhbmQgKGRldmljZS1hc3BlY3QtcmF0aW86IDIvMykge1xuICBAbWVkaWEgKG1heC13aWR0aDogNTAwcHgpIHtcbiAgICAuZm9ybVJlc2V0X2Vycm9ycyB7XG4gICAgICB0b3A6IDY1JTtcbiAgICB9XG4gICAgLmZvcm1SZXNldF9maWVsZCB7XG4gICAgICBwYWRkaW5nOiAzJSAwO1xuICAgIH1cbiAgfVxufVxuXG4vKipcbiAqIE1PVkUgRE9XTiBFRkZFQ1QgT04gU0NST0xMXG4gKi9cblxuLm1vdmVfZG93biB7XG4gIC1tcy10cmFuc2Zvcm06IHRyYW5zbGF0ZVkoMzJweCk7XG4gIC13ZWJraXQtdHJhbnNmb3JtOiB0cmFuc2xhdGVZKDMycHgpO1xuICAtbW96LXRyYW5zZm9ybTogdHJhbnNsYXRlWSgzMnB4KTtcbiAgLW8tdHJhbnNmb3JtOiB0cmFuc2xhdGVZKDMycHgpO1xuICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkoMzJweCk7XG59XG5cbi8qXG4gIE1BUktFUiBBTkQgUE9QVVAgTUFSS0VSXG4qL1xuXG4ubWFya2VyU21hcnRDaXRpemVuTm9ybWFsIHtcbiAgYm9yZGVyOiAxcHggc29saWQgJHllbGxvdztcbiAgYm9yZGVyLXJhZGl1czogMjBweDtcbiAgYm94LXNoYWRvdzogMHB4IDJweCA0cHggMHB4IHJnYmEoMCwwLDAsMC4zMCk7XG59XG5cbi5tYXJrZXJTbWFydENpdGl6ZW5PbmxpbmUge1xuICBib3JkZXI6IDJweCBzb2xpZCAkeWVsbG93O1xuICBib3JkZXItcmFkaXVzOiAyMHB4O1xuICBib3gtc2hhZG93OiAwcHggMnB4IDRweCAwcHggcmdiYSgwLDAsMCwwLjMwKTtcbiAgYmFja2dyb3VuZC1jb2xvcjogcmdiYSgkeWVsbG93LCAwLjUpO1xufVxuXG4ubWFya2VyRXhwZXJpbWVudGFsTm9ybWFsIHtcbiAgYm9yZGVyOiAycHggc29saWQgJGJhY2tncm91bmRfYmx1ZV9taWR0b25lO1xuICBib3JkZXItcmFkaXVzOiAyMHB4O1xuICBib3gtc2hhZG93OiAwcHggMnB4IDRweCAwcHggcmdiYSgwLDAsMCwwLjMwKTtcbiAgYmFja2dyb3VuZC1jb2xvcjogJGJhY2tncm91bmRfYmx1ZV9taWR0b25lO1xufVxuXG4vKiBAa2V5ZnJhbWVzIHB1bHNlIHtcbiAgMCUge1xuICAgIG9wYWNpdHk6IDAuMjtcbiAgfVxuICA1MCUge1xuICAgIG9wYWNpdHk6IDE7XG4gIH1cbiAgMTAwJSB7XG4gICAgb3BhY2l0eTogMC4yO1xuICB9XG59XG5ALXdlYmtpdC1rZXlmcmFtZXMgcHVsc2Uge1xuICAwJSB7XG4gICAgb3BhY2l0eTogMC4yO1xuICB9XG4gIDUwJSB7XG4gICAgb3BhY2l0eTogMTtcbiAgfVxuICAxMDAlIHtcbiAgICBvcGFjaXR5OiAwLjI7XG4gIH1cbn1cbkAtbW96LWtleWZyYW1lcyBwdWxzZSB7XG4gIDAlIHtcbiAgICBvcGFjaXR5OiAwLjI7XG4gIH1cbiAgNTAlIHtcbiAgICBvcGFjaXR5OiAxO1xuICB9XG4gIDEwMCUge1xuICAgIG9wYWNpdHk6IDAuMjtcbiAgfVxufVxuQC1vLWtleWZyYW1lcyBwdWxzZSB7XG4gIDAlIHtcbiAgICBvcGFjaXR5OiAwLjI7XG4gIH1cbiAgNTAlIHtcbiAgICBvcGFjaXR5OiAxO1xuICB9XG4gIDEwMCUge1xuICAgIG9wYWNpdHk6IDAuMjtcbiAgfVxufSAqL1xuXG4ubWFya2VyU21hcnRDaXRpemVuT2ZmbGluZSB7XG4gIGJvcmRlcjogMnB4IHNvbGlkIHJnYmEoMCwwLDAsIDAuMik7XG4gIGJvcmRlci1yYWRpdXM6IDIwcHg7XG4gIGJveC1zaGFkb3c6IDBweCAycHggNHB4IDBweCByZ2JhKDAsMCwwLDAuMzApO1xuICBiYWNrZ3JvdW5kLWNvbG9yOiByZ2JhKDAsMCwwLDAuMik7XG59XG5cbi8qIC5tYXJrZXJfYmxpbmsge1xuICAtd2Via2l0LWFuaW1hdGlvbjogcHVsc2UgMnMgbGluZWFyIGluZmluaXRlO1xuICAtbW96LWFuaW1hdGlvbjogcHVsc2UgMnMgbGluZWFyIGluZmluaXRlO1xuICAtbXMtYW5pbWF0aW9uOiBwdWxzZSAycyBsaW5lYXIgaW5maW5pdGU7XG4gIGFuaW1hdGlvbjogcHVsc2UgMnMgbGluZWFyIGluZmluaXRlO1xufSAqL1xuXG4ucG9wdXAge1xuICB3aWR0aDogMTAxJTs7XG4gIGhlaWdodDogMTM1cHg7XG4gIGNvbG9yOiAkb2ZmX2JsYWNrO1xuICBmb250LWZhbWlseTogJ1JvYm90byBDb25kZW5zZWQnXG59XG5cbi5wb3B1cF90b3Age1xuICBoZWlnaHQ6IDcycHg7XG4gIGJvcmRlci1yYWRpdXM6IDJweCAycHggMCAwO1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2xpZ2h0O1xuICBwYWRkaW5nLXRvcDo1cHg7XG5cbiAgYSB7XG4gICAgY29sb3I6ICRvZmZfYmxhY2s7XG4gICAgJjpob3ZlciB7XG4gICAgICBjb2xvcjogJGdyZXlfZGFya2VyO1xuICAgIH1cbiAgfVxuXG4gIC5wb3B1cF9uYW1lIHtcbiAgICBwYWRkaW5nOiAwIDAgMXB4IDhweDtcbiAgICBmb250LXNpemU6IDE2cHg7XG4gICAgZm9udC13ZWlnaHQ6IGJvbGQ7XG4gIH1cbiAgLnBvcHVwX3R5cGUge1xuICAgIGZvbnQtc2l6ZTogMTRweDtcbiAgICBsaW5lLWhlaWdodDogMWVtO1xuICAgIHBhZGRpbmc6IDFweCAwIDNweCA4cHg7XG4gIH1cbiAgLnBvcHVwX3RpbWUge1xuICAgIGZvbnQtc2l6ZTogMTFweDtcbiAgICBwYWRkaW5nOiAycHggMCAwIDhweDtcblxuICAgIG1kLWljb24ge1xuICAgICAgd2lkdGg6IDEwcHg7XG4gICAgICBoZWlnaHQ6IDEwcHg7XG4gICAgICBtYXJnaW4tcmlnaHQ6IDVweDtcbiAgICB9XG4gIH1cbn1cblxuLnBvcHVwX2JvdHRvbSB7XG4gIGhlaWdodDogNjhweDtcblxuICAucG9wdXBfbG9jYXRpb24ge1xuICAgIHBhZGRpbmc6IDVweCAwIDAgOHB4O1xuICAgIGZvbnQtc2l6ZTogMTRweDtcblxuICAgIG1kLWljb24ge1xuICAgICAgd2lkdGg6IDguNHB4O1xuICAgICAgaGVpZ2h0OiAxMnB4O1xuICAgICAgbWFyZ2luLXJpZ2h0OiA0cHg7XG4gICAgfVxuICB9XG4gIC5wb3B1cF9sYWJlbHMge1xuICAgIHBhZGRpbmc6IDhweCAwIDAgOHB4O1xuICAgIGZvbnQtc2l6ZTogMTJweDtcblxuICAgIC8vIHNwYW4ge1xuICAgIC8vICAgLy8gTWFrZSBiYWxsb29ucyBvbiBtYXAgc21hbGxlclxuICAgIC8vICAgLy9ib3JkZXItcmFkaXVzOiAyMHB4O1xuICAgIC8vICAgLy9tYXJnaW4tcmlnaHQ6IDEwcHg7XG4gICAgLy8gICAvL3BhZGRpbmc6IDNweCA4cHg7XG4gICAgLy8gfVxuXG4gICAgdGFnLCBzcGFuIHtcbiAgICAgIG1hcmdpbi1ib3R0b206IDhweDtcbiAgICAgIGZsb2F0OiBsZWZ0O1xuICAgIH1cbiAgfVxufVxuLnBvcHVwX2ljb24gc3ZnOm50aC1jaGlsZCgyKSB7XG4gIHdpZHRoOiAwO1xuICBoZWlnaHQ6IDA7XG59XG5cbi8qIEtpdCB0eXBlIGNvbG9ycyBraXRVdGlscy5zZXJ2aWNlLmpzIDo6IGNsYXNzaWZ5KCkqL1xuXG4uc2NrIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJHllbGxvdztcbn1cbi51bmtub3duIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogZ3JleTtcbn1cbi8qKlxuICogT1ZFUlJJREUgTEVBRkxFVFxuICovXG5cbi5sZWFmbGV0LWxlZnQge1xuICAvLyBUaGlzIGlzIGEgaGFjay4gVGhlIHJpZ2h0IHdheSBpcyB0byBjaGFuZ2UgdGhlIGxlYWZsZXQgQVBJLiBTZWUgdGhlIGRvY3M6XG4gIC8vIGh0dHA6Ly9sZWFmbGV0anMuY29tL3JlZmVyZW5jZS0wLjcuNy5odG1sI2NvbnRyb2wtcG9zaXRpb25cbiAgbGVmdDogdW5zZXQgIWltcG9ydGFudDtcbiAgcmlnaHQ6IDExcHg7XG4gIHRvcDogMTVweCAhaW1wb3J0YW50O1xufVxuXG4ubGVhZmxldC1wb3B1cC1jb250ZW50LXdyYXBwZXIge1xuICBib3JkZXItcmFkaXVzOiA0cHg7XG4gIHdpZHRoOiBhdXRvICFpbXBvcnRhbnQ7XG4gIG92ZXJmbG93OiBoaWRkZW47XG59XG5cbi5sZWFmbGV0LXBvcHVwLWNvbnRlbnQge1xuICBtYXJnaW46IDA7XG4gIG1pbi13aWR0aDogMjQ1cHg7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgdG9wOiAtMXB4O1xuICBsZWZ0OiAtMXB4O1xuXG4gIHAge1xuICAgIG1hcmdpbjogMDtcbiAgfVxufVxuXG4ubGVhZmxldC1wb3B1cC1jbG9zZS1idXR0b24ge1xuICBkaXNwbGF5OiBub25lO1xufVxuXG4ubGVhZmxldC1vdmVybGF5LXBhbmUgLmxlYWZsZXQtem9vbS1hbmltYXRlZCBwYXRoIHtcbiAgc3Ryb2tlOiBncmV5O1xufVxuLy8gb3ZlcnJpZGUgei1pbmRleCBvZiBtYXAgY29udHJvbHNcbi5sZWFmbGV0LXRvcC5sZWFmbGV0LWxlZnQge1xuICB6LWluZGV4OiAxO1xufVxuLmxlYWZsZXQtYm90dG9tLmxlYWZsZXQtcmlnaHQge1xuICB6LWluZGV4OiAxO1xufVxuXG4ubWFya2VyLWNsdXN0ZXIge1xuICBib3gtc2hhZG93OiAwcHggMnB4IDRweCAwcHggcmdiYSgwLDAsMCwwLjIwKTtcbiAgZGl2IHtcbiAgICAvL21hcmdpbi10b3A6IC0ycHg7XG4gICAgLy9tYXJnaW4tbGVmdDogLTJweDtcbiAgICAvL3dpZHRoOiA0MHB4O1xuICAgIC8vaGVpZ2h0OiA0MHB4O1xuICAgIGZvbnQtZmFtaWx5OiAnUm9ib3RvIENvbmRlbnNlZCc7XG4gICAgZm9udC1zaXplOiAxNHB4O1xuICAgIC8vbGluZS1oZWlnaHQ6IDQwcHg7XG4gICAgLy9mb250LXdlaWdodDogbm9ybWFsO1xuICB9XG59XG5cbi5tYXJrZXItY2x1c3Rlci1zbWFsbCwgLm1hcmtlci1jbHVzdGVyLW1lZGl1bSwgLm1hcmtlci1jbHVzdGVyLWxhcmdlIHtcbiAgICBib3JkZXItcmFkaXVzOiA1MCU7XG4gICAgYm9yZGVyOiAycHggc29saWQgcmdiYSgkeWVsbG93LCAwLjgpO1xuICAgIGJhY2tncm91bmQtY29sb3I6IHJnYmEoJHllbGxvdywgMC4zKTtcbiAgZGl2IHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiByZ2JhKDAsMjU1LDE2OCwwKTtcbiAgfVxufVxuXG4vKlxuICBQSUNLRVIgQ09NUE9ORU5UXG4qL1xuXG4ucGlja2VyIHtcbiAgY29sb3I6ICRvZmZfYmxhY2s7XG59XG5cbi5waWNrZXJfX3dlZWtkYXkge1xuICBjb2xvcjogJGdyZXlfZGFya2VyO1xufVxuXG4ucGlja2VyX19uYXYtLXByZXY6OmJlZm9yZSB7XG4gIGJvcmRlci1yaWdodDogMC43NWVtIHNvbGlkICRncmV5X2Rhcmtlcjtcbn1cblxuLnBpY2tlcl9fbmF2LS1uZXh0OjpiZWZvcmUge1xuICBib3JkZXItbGVmdDogMC43NWVtIHNvbGlkICRncmV5X2Rhcmtlcjtcbn1cblxuLnBpY2tlcl9fYnV0dG9uLS10b2RheTo6YmVmb3JlIHtcbiAgICBib3JkZXItdG9wOiAwLjY2ZW0gc29saWQgJGJsdWU7XG59XG5cbi5waWNrZXJfX2hvbGRlci5waWNrZXJfY29udGFpbmVyIHtcbiAgYmFja2dyb3VuZDogbm9uZTtcbiAgdHJhbnNpdGlvbjogbm9uZTtcbn1cblxuLnBpY2tlcl9fYnV0dG9uLS1jbGVhcjo6YmVmb3JlIHtcbiAgICBib3JkZXItdG9wOiAzcHggc29saWQgJHJlZDtcbn1cbi5waWNrZXJfX2J1dHRvbi0tY2xvc2Uge1xuICBjb2xvcjogJG9mZl9ibGFjaztcbn1cbi5waWNrZXJfX2J1dHRvbi0tdG9kYXksIC5waWNrZXJfX2J1dHRvbi0tY2xlYXIge1xuICBjb2xvcjogJG9mZl9ibGFjaztcbn1cbi5waWNrZXJfX2RheS0taGlnaGxpZ2h0ZWQsIC5waWNrZXJfX2RheS0tc2VsZWN0ZWQsIC5waWNrZXJfX2RheS0tc2VsZWN0ZWQ6aG92ZXIsIC5waWNrZXItLWZvY3VzZWQgLnBpY2tlcl9fZGF5LS1zZWxlY3RlZHtcbiAgYmFja2dyb3VuZDogJHllbGxvdztcbiAgYmFja2dyb3VuZC1jb2xvcjogJHllbGxvdztcbiAgYm9yZGVyLWNvbG9yOiAkeWVsbG93O1xuICBjb2xvcjogJG9mZl9ibGFjaztcbn1cbi5waWNrZXJfX2RheS0taW5mb2N1czpob3ZlciwgLnBpY2tlcl9fbmF2LS1wcmV2OmhvdmVyLCAucGlja2VyX19uYXYtLW5leHQ6aG92ZXIsIC5waWNrZXJfX2J1dHRvbi0tdG9kYXk6aG92ZXIsIC5waWNrZXJfX2J1dHRvbi0tY2xlYXI6aG92ZXIsIC5waWNrZXJfX2J1dHRvbi0tY2xvc2U6aG92ZXIge1xuICBiYWNrZ3JvdW5kOiAkeWVsbG93X2xpZ2h0ZXI7XG4gIGJvcmRlci1jb2xvcjogJHllbGxvd19saWdodGVyO1xufVxuXG4uZGF0ZV9waWNrZXIge1xuICBoZWlnaHQ6IDI0cHg7XG4gIGJvcmRlci1yYWRpdXM6IDJweDtcbiAgYm9yZGVyOiAxcHggc29saWQgJGdyZXlfZGFya2VyO1xuICBjb2xvcjogJGdyZXlfZGFya2VyO1xuICBiYWNrZ3JvdW5kLWNvbG9yOiB0cmFuc3BhcmVudDtcbn1cblxuLnNlbGVjdF9pbWFnZSB7XG4gIG1hcmdpbi1yaWdodDogMTBweDtcbn1cblxuLyoqXG4gKiBEUk9QRE9XTiBNRU5VXG4gKiBPdmVycmlkZSBzdHlsZSBmb3IgZHJvcGRvd24gbWVudSBmcm9tIGFuZ3VsYXItZHJvcGRvd25zIGRlcGVuZGVuY3lcbiAqL1xuIC53cmFwLWRkLW1lbnUgLmRyb3Bkb3duIGxpIGEge1xuICAgY29sb3I6ICRzZWNvbmRhcnlfY29sb3I7XG4gfVxuXG4ua2l0X21lbnUge1xuXG4gIC53cmFwLWRkLW1lbnUge1xuICAgIHdpZHRoOiAzMHB4O1xuICAgIGRpc3BsYXk6IGlubGluZS10YWJsZTtcbiAgICBwYWRkaW5nOiAwIDA7XG4gIH1cbiAgLndyYXAtZGQtbWVudSAuZHJvcGRvd24ge1xuICAgIHRvcDogNzAlO1xuICAgIGxpbmUtaGVpZ2h0OiAxZW07XG4gICAgbGVmdDogLTI1cHg7XG4gICAgcmlnaHQ6IC0yNXB4O1xuICAgIGxpIGEge1xuICAgICAgdGV4dC1hbGlnbjogY2VudGVyO1xuICAgICAgYm9yZGVyLWJvdHRvbTogMHB4O1xuICAgIH1cbiAgfVxufVxuLmtpdExpc3RfcGFyZW50IHtcblxuICAud3JhcC1kZC1tZW51IHtcbiAgICB3aWR0aDogNTBweDtcbiAgICBkaXNwbGF5OiBpbmxpbmUtYmxvY2s7XG4gIH1cbiAgLndyYXAtZGQtbWVudSAuZHJvcGRvd24ge1xuICAgIHRvcDogNzAlO1xuICAgIGxpbmUtaGVpZ2h0OiAxZW07XG4gICAgbGVmdDogLTI1cHg7XG4gICAgcmlnaHQ6IC0yNXB4O1xuICAgIGJvcmRlci1yYWRpdXM6IDBweDtcbiAgICBsaSBhIHtcbiAgICAgIHRleHQtYWxpZ246IGNlbnRlcjtcbiAgICAgIGJvcmRlci1ib3R0b206IDBweDtcbiAgICB9XG4gIH1cbn1cblxuLmtpdF9vd25lcl9kcm9wZG93bl9idXR0b24ge1xuICBkaXNwbGF5OiBibG9jaztcbiAgbWFyZ2luOiAwIGF1dG87XG59XG5cbi8qKlxuICogIEtJVCBMSVNUXG4gKlxuICovXG4ua2l0TGlzdCB7XG4gIG1hcmdpbjogMHB4O1xuICBjb2xvcjogJGdyZXk7XG4gIGJvcmRlci10b3A6IDFweCBzb2xpZCAjRUFFQ0YxO1xuICBib3JkZXItcmlnaHQ6IDFweCBzb2xpZCAjRUFFQ0YxO1xuICBib3JkZXItbGVmdDogMXB4IHNvbGlkICNFQUVDRjE7XG4gIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xuICBib3JkZXItcmFkaXVzOiAwICFpbXBvcnRhbnQ7XG4gIHRleHQtdHJhbnNmb3JtOiBub25lO1xuICBwYWRkaW5nOjE1cHg7XG5cbiAgaDEsaDIsaDMsaDR7XG4gICAgbGluZS1oZWlnaHQ6IGluaGVyaXQ7XG4gIH1cbiAgcHtcbiAgICBsaW5lLWhlaWdodDogMC40ZW07XG4gIH1cblxuICAud3JhcC1kZC1tZW51IHtcbiAgICB3aWR0aDogNTBweDtcbiAgICBkaXNwbGF5OiBpbmxpbmUtYmxvY2s7XG4gIH1cbiAgLndyYXAtZGQtbWVudSAuZHJvcGRvd24ge1xuICAgIHRvcDogNzAlO1xuICAgIGxpbmUtaGVpZ2h0OiAxZW07XG4gICAgbGVmdDogLTI1cHg7XG4gICAgcmlnaHQ6IC0yNXB4O1xuICAgIGJvcmRlci1yYWRpdXM6IDBweDtcbiAgICBsaSBhIHtcbiAgICAgIHRleHQtYWxpZ246IGNlbnRlcjtcbiAgICAgIGJvcmRlci1ib3R0b206IDBweDtcbiAgICB9XG4gIH1cblxuICAuY2hpcHMge1xuICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICByaWdodDogMTMwcHg7XG4gIH1cblxufVxuXG4ua2l0TGlzdF9hdmF0YXIge1xuICB3aWR0aDogNDRweDtcbiAgaGVpZ2h0OiA0NHB4O1xuICBib3JkZXItcmFkaXVzOiAyMnB4O1xufVxuLmtpdExpc3RfbmFtZSB7XG4gIG1hcmdpbi1ib3R0b206IDVweDtcbiAgZm9udC1zaXplOiAxOHB4O1xuICBjb2xvcjogJGJsdWU7XG59XG4ua2l0TGlzdF9yaWdodCB7XG4gIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgcmlnaHQ6IDA7XG5cbiAgLmNoaXAgLmNoaXBfbmFtZSB7XG4gICAgdGV4dC10cmFuc2Zvcm06IHVwcGVyY2FzZTtcbiAgfVxufVxuLmtpdExpc3Rfc3RhdGUge1xuICBjb2xvcjp3aGl0ZTtcbiAgdGV4dC10cmFuc2Zvcm06IHVwcGVyY2FzZTtcbn1cbi5raXRMaXN0X3N0YXRlX2hhc19wdWJsaXNoZWQge1xuICBjb2xvcjogJGdyZWVuO1xufVxuLmtpdExpc3Rfc3RhdGVfbmV2ZXJfcHVibGlzaGVkIHtcbiAgY29sb3I6ICR5ZWxsb3c7XG59XG4ua2l0TGlzdF9zdGF0ZV9ub3RfY29uZmlndXJlZCB7XG4gIGNvbG9yOiAjRjQzRDREO1xufVxuLmtpdExpc3RfZHJvcGRvd25CdXR0b24ge1xuICBkaXNwbGF5OiBibG9jaztcbiAgbWFyZ2luOiAwIGF1dG87XG5cbiAgaW1nIHtcbiAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gIH1cbn1cblxuLmtpdExpc3Rfbm9LaXRzIHtcbiAgbGluZS1oZWlnaHQ6IDU7XG59XG5cbi5kcm9wZG93bi1pdGVtLWJ1dHRvbiB7XG4gICAgYm9yZGVyLXJhZGl1czogNXB4O1xuICAgIHRleHQtYWxpZ246IGNlbnRlcjtcbiAgICB3aWR0aDogMTAwJTtcbn1cbi5kcm9wZG93bi1pdGVtLWJ1dHRvbjpob3ZlcntcbiAgYmFja2dyb3VuZC1jb2xvcjogJHllbGxvdyAhaW1wb3J0YW50XG59XG5cbm1kLWNhcmQua2l0LWxpc3QtaXRlbSB7XG4gIGJvcmRlci1yYWRpdXM6IDEwcHg7XG59XG5cbi8qKlxuICogIFVTRVIgUFJPRklMRSBTVEFURVxuICogIG1vc3Qgc3R5bGVzIGFyZSByZS11c2VkIG9uIG15IHByb2ZpbGUgc3RhdGVcbiAqL1xuXG4vKiBwcm9maWxlIGhlYWRlciAqL1xuLnByb2ZpbGVfaGVhZGVyIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJGJhY2tncm91bmRfYmx1ZV9saWdodGVyX21lbnU7XG4gIGNvbG9yOiAkZm9udF9jb2xvcl9ibHVlX2dyZXk7XG4gIGhlaWdodDogMTkycHg7XG4gIG1hcmdpbi10b3A6IDY0cHg7XG59XG4ucHJvZmlsZV9oZWFkZXJfYXZhdGFyIHtcbiAgd2lkdGg6IDEwMHB4O1xuICBoZWlnaHQ6IDEwMHB4O1xuICBib3JkZXItcmFkaXVzOiA1MHB4O1xuICBtYXJnaW4tcmlnaHQ6IDI5cHg7XG4gIG1hcmdpbi1sZWZ0OiAyN3B4O1xufVxuLnByb2ZpbGVfaGVhZGVyX2NvbnRlbnQge1xuICA+ZGl2IHtcbiAgICBtYXJnaW46IDVweCAwO1xuICB9XG4gIHN2ZyB7XG4gICAgZmlsbDogJGdyZXlfbGlnaHRlcjtcbiAgfVxufVxuLnByb2ZpbGVfaGVhZGVyX2NvbnRlbnRfYXZhdGFyIHtcbiAgd2lkdGg6IDEycHg7XG4gIGhlaWdodDogMTJweDtcbiAgbWFyZ2luLXJpZ2h0OiA2cHg7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgdG9wOiAtMXB4O1xufVxuXG4ucHJvZmlsZV9jb250ZW50IHtcbiAgbWluLWhlaWdodDogNzAwcHg7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbn1cbi8qIHByb2ZpbGUgc2lkZWJhciAqL1xuLnByb2ZpbGVfc2lkZWJhciB7XG4gIC8vbWluLWhlaWdodDogNDAwcHg7XG4gIC8vIG1pbi1oZWlnaHQ6IGluaGVyaXQ7XG4gIC8vIHdpZHRoOiAyNTZweDtcbiAgYmFja2dyb3VuZC1jb2xvcjogI0Y1RjVGNTtcbiAgbWluLXdpZHRoOjIwMHB4O1xufVxuLnByb2ZpbGVfc2lkZWJhcl90aXRsZSB7XG4gIGNvbG9yOiAkb2ZmX2JsYWNrO1xuICBmb250LXdlaWdodDogYm9sZDtcbn1cbi5wcm9maWxlX3NpZGViYXJfZGVzY3JpcHRpb24ge1xuICBmb250LXNpemU6IDE0cHg7XG59XG5cbi5wcm9maWxlX3NpZGViYXJfYnV0dG9uIHtcbiAgaGVpZ2h0OiA2NHB4O1xuICBjb2xvcjogJHRlcmNpYXJ5X2NvbG9yO1xuICBzdmcgeyBmaWxsOiAkYmx1ZTsgfVxufVxuXG4ucHJvZmlsZV9zaWRlYmFyX2F2YXRhciB7XG4gIHdpZHRoOiAxMnB4O1xuICBoZWlnaHQ6IDEycHg7XG4gIG1hcmdpbi1yaWdodDogOHB4O1xuICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gIHRvcDogLTJweDtcbn1cblxuLyogcHJvZmlsZSBtYWluIGNvbnRlbnQgKi9cbi5wcm9maWxlX2NvbnRlbnRfbWFpbl90b3Age1xuICBtYXJnaW46IDAgMCA0MnB4IDIwcHg7XG59XG5cbi8qKlxuICogIE1ZIFBST0ZJTEUgU1RBVEVcbiAqL1xuXG4vL292ZXJyaWRlIGFuZ3VsYXItbWF0ZXJpYWxcbi5teVByb2ZpbGVfc3RhdGUge1xuICBtZC1pbmstYmFye1xuICAgIC8vIFRhYnMgdW5kZXJsaW5lIGNvbG9yIGN5YW5cbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkdGVyY2lhcnlfY29sb3I7XG4gIH1cbiAgbWQtdGFiIG1kLXRhYi1sYWJlbCB7XG4gICAgb3ZlcmZsb3c6IHZpc2libGU7XG4gIH1cbiAgbWQtdGFicyBtZC10YWIge1xuICAgIGNvbG9yOiAkZm9udF9jb2xvcl9ibHVlX2dyZXk7XG4gIH1cbiAgbWQtdGFicyBtZC10YWIuYWN0aXZlIHtcbiAgICBjb2xvcjogd2hpdGU7XG4gIH1cbiAgbWQtdGFicyBtZC10YWJzLWluay1iYXIge1xuICAgIGNvbG9yOiAkYmx1ZTtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmx1ZTtcbiAgfVxufVxuXG5tZC10YWJzLWNhbnZhcyB7XG4gIGJhY2tncm91bmQtY29sb3I6ICMyRTJFMkU7XG59XG5cbi5teVByb2ZpbGVfaGVhZGVyIHtcbiAgaGVpZ2h0OiAxODhweDtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xufVxuLm15UHJvZmlsZV9oZWFkZXJfY29udGFpbmVyIHtcbiAgcGFkZGluZy10b3A6IDUxcHg7XG4gIG1hcmdpbi1sZWZ0OiA0NnB4O1xufVxuLm15UHJvZmlsZV9oZWFkZXJfYXZhdGFyIHtcbiAgbWFyZ2luLWxlZnQ6IDA7XG59XG4ubXlQcm9maWxlX3RhYnNfcGFyZW50IHtcbiAgbWQtdGFicy13cmFwcGVye1xuICAgIGJhY2tncm91bmQtY29sb3I6ICRiYWNrZ3JvdW5kX2JsdWVfZGFya19tZW51O1xuICB9XG59XG4vLyAubXlQcm9maWxlX3RhYnMge1xuLy8gfVxuLm15UHJvZmlsZV90YWJfaWNvbiB7XG4gIHdpZHRoOiAxNnB4O1xuICBoZWlnaHQ6IDE2cHg7XG4gIG1hcmdpbi1yaWdodDogOHB4O1xuICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gIHRvcDogLTFweDtcbiAgc3ZnIHtcbiAgICBmaWxsOiB3aGl0ZTtcbiAgfVxufVxuLm15UHJvZmlsZV9zaWRlYmFyX2J1dHRvbiB7XG4gIHBhZGRpbmctbGVmdDogMjhweDtcbn1cbi5teVByb2ZpbGVfY29udGVudF9mb3JtIHtcbiAgbWFyZ2luLWxlZnQ6IDExM3B4O1xuICBtYXgtd2lkdGg6IDUwMHB4O1xufVxuLm15UHJvZmlsZV9jb250ZW50X2Zvcm1faW5wdXQge1xuICBtYXJnaW46IDE2cHggMDtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xufVxuLmNvdW50cnlJbnB1dF9jb250YWluZXIge1xuICBtZC1hdXRvY29tcGxldGUge1xuICAgIGJhY2tncm91bmQtY29sb3I6IHRyYW5zcGFyZW50O1xuICB9XG4gIG1kLWF1dG9jb21wbGV0ZS13cmFwIHtcbiAgICBib3gtc2hhZG93OiBub25lICFpbXBvcnRhbnQ7XG4gIH1cbiAgaW5wdXQge1xuICAgIGZvbnQtc2l6ZTogMTAwJTtcbiAgfVxuICBsYWJlbCB7XG4gICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgIGJvdHRvbTogNTVweDtcbiAgfVxufVxuLm15UHJvZmlsZV9mb3JtX2F2YXRhciB7XG4gIG1hcmdpbi1ib3R0b206IDMwcHg7XG59XG4ubXlQcm9maWxlX2Zvcm1fYXZhdGFySW1hZ2Uge1xuICB3aWR0aDogNjRweDtcbiAgaGVpZ2h0OiA2NHB4O1xuICBib3JkZXItcmFkaXVzOiAzMnB4O1xuICBtYXJnaW4tcmlnaHQ6IDclO1xufVxuLm15UHJvZmlsZV9hcGlLZXlfdGV4dCB7XG4gIGNvbG9yOiAkZm9udF9jb2xvcl9ibHVlX2dyZXk7XG4gIG1hcmdpbi1yaWdodDogNXB4O1xuICB3aWR0aDogMTIwcHg7XG59XG4ubXlQcm9maWxlX2FwaUtleV9udW1iZXIge1xuICBjb2xvcjogIzlEOUQ5RTtcbiAgZm9udC13ZWlnaHQ6IGJvbGQ7XG4gIGJhY2tncm91bmQtY29sb3I6ICNGMkYyRjI7XG4gIHBhZGRpbmc6IDVweCAxMHB4O1xufVxuLm15UHJvZmlsZV9hcGlLZXlfcmVmcmVzaEJ1dHRvbiB7XG4gIGJhY2tncm91bmQtY29sb3I6ICRmb250X2NvbG9yX2JsdWVfZ3JleTtcbiAgaGVpZ2h0OiAyOXB4O1xuXG4gIG1kLWljb24ge1xuICAgIGhlaWdodDogMTVweDtcbiAgICB3aWR0aDogMTVweDtcbiAgICBkaXNwbGF5OiBibG9jaztcbiAgICBtYXJnaW46IDNweCBhdXRvO1xuICB9XG59XG4ubWQtYnV0dG9uLm15UHJvZmlsZV9hcGlLZXlfcmVmcmVzaEJ1dHRvbiB7XG4gIGJvcmRlci1yYWRpdXM6IDAgNHB4IDRweCAwO1xufVxuLm1kLWJ1dHRvbi5teVByb2ZpbGVfYXBpS2V5X3JlZnJlc2hCdXR0b246YWN0aXZlLCAubWQtYnV0dG9uLm15UHJvZmlsZV9hcGlLZXlfcmVmcmVzaEJ1dHRvbjpob3ZlciwgLm1kLWJ1dHRvbi5teVByb2ZpbGVfYXBpS2V5X3JlZnJlc2hCdXR0b246Zm9jdXMge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkZm9udF9jb2xvcl9ibHVlX2dyZXk7XG59XG4ubXlQcm9maWxlX2Zvcm1fcmVtb3ZlQnV0dG9uIHtcbiAgZGlzcGxheTogYmxvY2s7XG4gIGNvbG9yOiAkYWxlcnRfcmVkO1xuICBtYXJnaW4tYm90dG9tOiA1cHg7XG59XG4ubXlQcm9maWxlX3VwZGF0ZUZvcm1fZXJyb3Ige1xuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHRvcDogNTBweDtcbiAgbGVmdDogMnB4O1xuICBjb2xvcjogcmVkO1xuICBmb250LXNpemU6IDEycHg7XG4gIGxpbmUtaGVpZ2h0OiAxO1xufVxuLm15UHJvZmlsZV9hcGlLZXlfYmxvY2sge1xuICBtYXJnaW4tdG9wOiA2M3B4O1xuICBtYXJnaW4tYm90dG9tOiA2NXB4O1xufVxuXG4ubXlQcm9maWxlX2FwaUtleSB7XG4gIG1hcmdpbi10b3A6IDEwcHg7XG4gIG1hcmdpbi1ib3R0b206IDEwcHg7XG4gIHN2ZyB7XG4gICAgZmlsbDogJGdyZXlfZGFya2VyO1xuICB9XG59XG5cbi5tZC1idXR0b24ubXlQcm9maWxlX2FkZEtpdEJ1dHRvbjpmb2N1cyB7XG4gIGJhY2tncm91bmQtY29sb3I6ICRibHVlO1xufVxuLm1kLWJ1dHRvbi5teVByb2ZpbGVfYWRkS2l0QnV0dG9uOmhvdmVyIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJGJsdWU7XG59XG4ubXlQcm9maWxlX2hpZGRlbmhyZWZ7XG4gIHBvc2l0aW9uOmFic29sdXRlO1xuICBoZWlnaHQ6IDEwMCU7XG4gIHdpZHRoOjEwMCU7XG4gIHRvcDowO1xuICBsZWZ0OjA7XG59XG5cbi8qKlxuICogVE9PTCBMSVNUXG4gKlxuICovXG5cbiBhLmtpdExpc3Qge1xuICAgaDQge1xuICAgICBjb2xvcjogJG9mZl9ibGFjaztcbiAgIH1cbiAgIHAge1xuICAgICBjb2xvcjogJGdyZXlfZGFya2VyO1xuICAgfVxuIH1cblxuLmNsZWFyZml4IHtcbiAgY2xlYXI6IGJvdGg7XG59XG5cbi8qXG4gIEZPT1RFUiBDT01QT05FTlRcbiovXG5cbmZvb3RlciB7XG4vLyAgcG9zaXRpb246IGFic29sdXRlO1xuICB6LWluZGV4OiAxO1xuICB3aWR0aDogMTAwJTtcbiAgYmFja2dyb3VuZC1jb2xvcjogJGJhY2tncm91bmRfYmx1ZV9kYXJrX2JhY2tncm91bmQ7XG59XG5cbi5mb290ZXJfaWNvbiB7XG4gIGRpc3BsYXk6IGJsb2NrO1xuICB3aWR0aDogMzJweDtcbiAgaGVpZ2h0OiAzMnB4O1xufVxuLmZsYWcge1xuICBwYWRkaW5nLXJpZ2h0OiAxNXB4O1xufVxuLmJvcmRlci13aGl0ZS5mb290ZXItYmxvY2t7XG4gIGJvcmRlcjogMnB4IHdoaXRlIHNvbGlkO1xufVxuLnVwdGltZXJvYm90LWxvZ28ge1xuICB3aWR0aDogMTUwcHg7XG4gIGp1c3RpZnktc2VsZjogcmlnaHQ7XG4gIHZlcnRpY2FsLWFsaWduOiBjZW50ZXI7XG4gIHBhZGRpbmctdG9wOiAxMHB4O1xufVxuLnVwdGltZXJvYm90LXNwb25zb3Ige1xuICBkaXNwbGF5OiBmbGV4O1xuICBmbGV4LWRpcmVjdGlvbjogY29sdW1uO1xuICBqdXN0aWZ5LWNvbnRlbnQ6IHN0YXJ0O1xuICBhbGlnbi1pdGVtczogZmxleC1zdGFydDtcbn1cbi51cHRpbWVyb2JvdC1zcG9uc29yLXRleHQgcCB7XG4gIGZvbnQtc2l6ZTogMTVweCAhaW1wb3J0YW50O1xuICBtYXJnaW46IDBweDtcbn1cbkBtZWRpYSBvbmx5IHNjcmVlbiBhbmQgKG1heC13aWR0aDogOTUwcHgpIHtcbiAgLmNvbG9yLXdoaXRle1xuICAgIHRleHQtYWxpZ246IGNlbnRlcjtcbiAgfVxuICAuZmxhZyB7XG4gICAgcGFkZGluZy1yaWdodDogMHB4O1xuICB9XG4gIC5mb290ZXItYmxvY2sge1xuICAgIG1hcmdpbi1ib3R0b206IDIwcHggIWltcG9ydGFudDtcbiAgfVxuICAuc3BvbnNvciB7XG4gICAganVzdGlmeS1jb250ZW50OiBjZW50ZXI7XG4gICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgfVxufVxuQG1lZGlhIChtYXgtd2lkdGg6IDU5OXB4KSB7XG4gIC5sYXlvdXQtYWxpZ24teHMtc3RhcnQtc3RyZXRjaCwgLmxheW91dC1hbGlnbi14cy1jZW50ZXItc3RyZXRjaCwgLmxheW91dC1hbGlnbi14cy1lbmQtc3RyZXRjaCwgLmxheW91dC1hbGlnbi14cy1zcGFjZS1iZXR3ZWVuLXN0cmV0Y2gsIC5sYXlvdXQtYWxpZ24teHMtc3BhY2UtYXJvdW5kLXN0cmV0Y2h7XG4gICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgfVxufVxuXG5cbi8qXG4gIFJFQ09WRVJZIFBBU1NXT1JEIFNUQVRFXG4qL1xuLnJlY292ZXJ5X2hlYWRlciB7XG4gIHdpZHRoOiAxMDAlO1xuICBoZWlnaHQ6IDEyOHB4O1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2RhcmtfYmFja2dyb3VuZDtcbn1cblxuLnJlY292ZXJ5X2hlYWRlckljb24ge1xuICBkaXNwbGF5OiBibG9jaztcbiAgd2lkdGg6IDUwcHg7XG4gIGhlaWdodDogNTBweDtcbn1cblxuLypcbiAgVElNRUxJTkUgQ09NUE9ORU5UXG5cbiAgdXNlZCBvbiBhZGQgYW5kIHNldHVwIGtpdFxuKi9cblxuLnRpbWVsaW5lIHtcbiAgLy8gbWFyZ2luLXRvcDogNjRweDtcbiAgLy8gYmFja2dyb3VuZC1jb2xvcjogJGJhY2tncm91bmRfYmx1ZV9kYXJrX21lbnU7XG4gIGhlaWdodDogJHRpbWVsaW5lLWhlaWdodDtcbiAgYmFja2dyb3VuZC1jb2xvcjogJGJhY2tncm91bmRfYmx1ZV9saWdodGVyX21lbnU7XG4gIHBvc2l0aW9uOiBmaXhlZDtcbiAgd2lkdGg6IDEwMCU7XG4gIHotaW5kZXg6IDEwO1xuICB0b3A6IDY0cHg7XG4gIGJveC1zaGFkb3c6IDAgNHB4IDVweCAtMnB4ICRiYWNrZ3JvdW5kX2JsdWVfbGlnaHRlcl9tZW51O1xuICBwYWRkaW5nOiAzMHB4O1xufVxuXG5AbWVkaWEgKG1heC13aWR0aDogOTYwcHgpIHtcbiAgLnRpbWVsaW5lLWJ0bi1leHRyYSB7XG4gICAgZGlzcGxheTogTm9uZTtcbiAgfVxufVxuXG5AbWVkaWEgKG1pbi13aWR0aDogMHB4KSBhbmQgKG1heC13aWR0aDogOTU5cHgpIGFuZCAob3JpZW50YXRpb246IHBvcnRyYWl0KSB7XG4gIC50aW1lbGluZSB7XG4gICAgdG9wOiA1NnB4O1xuICB9XG4gIC5wcm9maWxlX2hlYWRlciB7XG4gICAgbWFyZ2luLXRvcDogNTNweDtcbiAgfVxufVxuXG5AbWVkaWEgKG1pbi13aWR0aDogMHB4KSBhbmQgKG1heC13aWR0aDogOTU5cHgpIGFuZCAob3JpZW50YXRpb246IGxhbmRzY2FwZSkge1xuICAudGltZWxpbmUge1xuICAgIHRvcDogNTNweDtcbiAgfVxuICAucHJvZmlsZV9oZWFkZXIge1xuICAgIG1hcmdpbi10b3A6IDUzcHg7XG4gIH1cbn1cblxuLnRpbWVsaW5lX2NvbnRhaW5lciB7XG4gIHdpZHRoOiAxMDAlO1xufVxuLnRpbWVsaW5lX2xpbmUge1xuICB3aWR0aDogNTAwcHg7XG4gIGJvcmRlcjogMXB4IHNvbGlkICR5ZWxsb3c7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgdG9wOiAxNHB4O1xufVxuLnRpbWVsaW5lX2xpbmVfc21hbGwge1xuICB3aWR0aDogMTAwcHggIWltcG9ydGFudDtcbiAgbWFyZ2luOiAwcHggMjBweDtcbiAgdG9wOjBweCAhaW1wb3J0YW50O1xufVxuXG4udGltZWxpbmUtdGl0bGUge1xuICB0ZXh0LXRyYW5zZm9ybTogdXBwZXJjYXNlO1xuICBjb2xvcjogJHllbGxvdztcbiAgZm9udC1zaXplOjI0cHg7XG4gIGZvbnQtd2VpZ2h0OiA2MDA7IC8vIFRPRE86IENvc21ldGljXG5cbiAgJi52ZXJ0aWNhbCB7XG4gICAgbWFyZ2luLXRvcDogMTBweDtcbiAgfVxufVxuXG4udGltZWxpbmVfc3RlcE5hbWUge1xuICB0ZXh0LXRyYW5zZm9ybTogdXBwZXJjYXNlO1xuICBjb2xvcjogJHllbGxvdztcbiAgZm9udC1zaXplOjE4cHg7XG5cbiAgJi52ZXJ0aWNhbCB7XG4gICAgbWFyZ2luLXRvcDogMTBweDtcbiAgfVxufVxuXG4udGltZWxpbmVfY29udGVudCB7XG4gIG1pbi1oZWlnaHQ6IDQwMHB4O1xuICB3aWR0aDogMTAwJTtcbiAgbWFyZ2luLXRvcDogJGNvbnRlbnQtbWFyZ2luLXRvcDtcbn1cblxuLy8gLnRpbWVsaW5lX2J1dHRvbiB7XG4vLyAgIHdpZHRoOiA5OSU7XG4vLyAgIGhlaWdodDogOTBweDtcbi8vICAgZm9udC1zaXplOiAyNnB4O1xuXG4vLyAgICYudGltZWxpbmVfYnV0dG9uT3Blbi5tZC1wcmltYXJ5IHtcbi8vICAgICBjb2xvcjogJHRlcmNpYXJ5X2NvbG9yO1xuLy8gICAgIHRleHQtYWxpZ246IGNlbnRlcjtcblxuLy8gICAgICYuaW52ZXJ0ZWR7XG4vLyAgICAgICBjb2xvcjogd2hpdGU7XG4vLyAgICAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkdGVyY2lhcnlfY29sb3I7XG5cbi8vICAgICAgIHNtYWxsIHtcbi8vICAgICAgICAgY29sb3I6IHdoaXRlO1xuLy8gICAgICAgfVxuLy8gICAgIH1cblxuLy8gICAgIC5tYWluIHtcbi8vICAgICAgIG1hcmdpbjogMTBweDtcbi8vICAgICAgIGZvbnQtc2l6ZTogMzJweDtcbi8vICAgICB9XG5cbi8vICAgICAuc3ViIHtcbi8vICAgICAgIG1hcmdpbjogMTBweDtcbi8vICAgICAgIGZvbnQtc2l6ZTogMTZweDtcbi8vICAgICB9XG5cbi8vICAgICBzbWFsbCB7XG4vLyAgICAgICBjb2xvcjogcmdiYSgwLDAsMCwwLjU0KTtcbi8vICAgICAgIHRleHQtdHJhbnNmb3JtOiBpbml0aWFsO1xuLy8gICAgICAgZGlzcGxheTogYmxvY2s7XG4vLyAgICAgICBmb250LXNpemU6IDE2cHg7XG4vLyAgICAgfVxuXG4vLyAgIH1cbi8vIH1cblxuLm1kLWJ1dHRvbi50aW1lbGluZV9idXR0b25CYWNre1xuICBtYXJnaW4tbGVmdDogYXV0bztcbiAgbWFyZ2luLXJpZ2h0OiAyMHB4O1xufVxuXG5zZWN0aW9uLnJlbGF4ZWQtbGF5b3V0IHtcbiAgcGFkZGluZzogMTBweCAyNXB4O1xufVxuXG5cblxuLypcbiAgQUREIEtJVCBTVEFURVxuKi9cblxuLmtpdF9kYXRhQ2hhbmdlIHtcbiAgLmZvcm1fYmxvY2sge1xuICAgIHBhZGRpbmc6IDQwcHggMzBweDtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjRkZGO1xuXG4gICAgJi5pc0V2ZW4ge1xuICAgICAgYmFja2dyb3VuZC1jb2xvcjogI0Y5RjlGQjtcbiAgICB9XG4gICAgLy8gJi5mb3JtX2Jsb2NrTm9ybWFsIHtcbiAgICAvLyAgLy8gbWluLWhlaWdodDogMjUwcHg7XG4gICAgLy8gfVxuICAgIC8vICYuZm9ybV9ibG9ja01hcCB7XG4gICAgLy8gICAvL2hlaWdodDogMjkwcHg7XG4gICAgLy8gfVxuICB9XG4gIC5mb3JtX2Jsb2NrSW5wdXQge1xuICAgIGhlaWdodDogaW5oZXJpdDtcbiAgIC8vIG1hcmdpbi1sZWZ0OiAxNTBweDtcbiAgfVxuICAuZm9ybV9ibG9ja0lucHV0X2NvbnRhaW5lciB7XG4gICAgaGVpZ2h0OiBpbmhlcml0O1xuICB9XG4gIC5mb3JtX2Jsb2NrSW5wdXRfYnV0dG9uIHtcbiAgICBoZWlnaHQ6aW5oZXJpdDtcbiAgfVxuICAuZm9ybV9ibG9ja0lucHV0X21hcCB7XG4gICAgaGVpZ2h0OiAyNTBweDtcbiAgICBtaW4td2lkdGg6IDI1MHB4O1xuICB9XG4gIC5mb3JtX2Jsb2NrSW5wdXRfc2VsZWN0IGxhYmVse1xuICAgIGNvbG9yOiAkdGVyY2lhcnlfY29sb3I7XG4gICAgbWFyZ2luLXJpZ2h0OiAxMHB4O1xuICB9XG4gIC5mb3JtX2Jsb2NrSW5wdXRfY2hpcHMge1xuICAgIG1hcmdpbi10b3A6IDM1cHg7XG4gIH1cbiAgLmZvcm1fYmxvY2tDb250ZW50IHtcbiAgICBwYWRkaW5nLXRvcDogNXB4O1xuICAgIGZsb2F0OiBsZWZ0O1xuICAgIGltZywgZGl2IHtcbiAgICAgIGZsb2F0OiBsZWZ0O1xuICAgIH1cbiAgICBpbWcge1xuICAgICAgYm9yZGVyLXJhZGl1czogNTAlO1xuICAgICAgd2lkdGg6IDEyMHB4O1xuICAgICAgaGVpZ2h0OiAxMjBweDtcbiAgICAgIG1hcmdpbjogNXB4IDIwcHg7XG4gICAgfVxuICB9XG4gIC5mb3JtX2Jsb2NrQ29udGVudF9pbWFnZSB7XG4gICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgIHRvcDogM3B4O1xuICB9XG4gIC5mb3JtX2Jsb2NrQ29udGVudF90ZXh0IHtcbiAgICBtYXJnaW4tbGVmdDogMTBweDtcbiAgICB3aWR0aDogMjQwcHg7XG4gICAgJi5sb25ne1xuICAgICAgd2lkdGg6IGF1dG8gIWltcG9ydGFudDtcbiAgICB9XG4gICAgaDIge1xuICAgICAgbWFyZ2luLXRvcDogMDtcbiAgICAgIG1hcmdpbi1ib3R0b206IDhweDtcbiAgICB9XG4gICAgcCB7XG4gICAgICBmb250LXNpemU6IDE0cHg7XG4gICAgICBjb2xvcjogI0E1QTVBNTtcbiAgICAgIGxpbmUtaGVpZ2h0OiAxLjU7XG4gICAgfVxuICB9XG4gIC5mb3JtX2Vycm9ycyB7XG4gICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgIGJvdHRvbTogMTUlO1xuICAgIGxlZnQ6IDA7XG4gICAgY29sb3I6IHJlZDtcbiAgICBmb250LXNpemU6IDAuN2VtO1xuICAgIGxpbmUtaGVpZ2h0OiAxO1xuICB9XG59XG5cbi5lbW9qaXtcbiAgZm9udC13ZWlnaHQ6NDAwO1xufVxuXG4vKlxuICBTVEFUSUMgUEFHRVNcbiovXG4uc3RhdGljX3BhZ2V7XG4gIC50aW1lbGluZXtcbiAgICBhbGlnbi1pdGVtczogZmxleC1zdGFydCBzdHJldGNoO1xuXG4gICAgLmNvbnRlbnR7XG4gICAgICBwYWRkaW5nOjAgMHB4O1xuICAgIH1cbiAgfVxuXG4gIC5jb250ZW50e1xuICAgIG1heC13aWR0aDogNTBlbTtcbiAgICBtYXJnaW46IDAgYXV0bztcbiAgICB0ZXh0LWFsaWduOiBqdXN0aWZ5O1xuICAgIHBhZGRpbmc6NDBweCAxMHB4O1xuICB9XG5cbiAgLmZ1bGwtd2lkdGgtaW1nIHtcbiAgICB3aWR0aDogMTAwJTtcbiAgICBoZWlnaHQ6IGF1dG87XG4gICAgbWF4LXdpZHRoOiAyMDAwcHg7XG4gICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgIG1hcmdpbjogMCAwIC01cHggMDtcbiAgfVxuXG4gIC5lbWJlZC1jb250YWluZXIge1xuICAgICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgICAgcGFkZGluZy1ib3R0b206IDU2LjI1JTtcbiAgICAgIGhlaWdodDogMDtcbiAgICAgIG92ZXJmbG93OiBoaWRkZW47XG4gICAgICBtYXgtd2lkdGg6IDEwMCU7XG4gICAgICBpZnJhbWUsIG9iamVjdCwgZW1iZWQge1xuICAgICAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgICAgIHRvcDogMDtcbiAgICAgICAgbGVmdDogMDtcbiAgICAgICAgd2lkdGg6IDEwMCU7XG4gICAgICAgIGhlaWdodDogMTAwJTtcbiAgICAgIH1cbiAgfVxuXG4gIC5jZW50ZXIge1xuICAgICAgdGV4dC1hbGlnbjogY2VudGVyO1xuICAgICAgbWFyZ2luLXRvcDogMzBweDtcbiAgfVxuXG4gIC5yb3d7XG4gICAgbWluLWhlaWdodDogMjAwcHg7XG5cbiAgICAmOm50aC1jaGlsZChvZGQpe1xuICAgICAgYmFja2dyb3VuZC1jb2xvcjogI0Y5RjlGQjtcbiAgICB9XG4gIH1cbiAgaDF7XG4gICAgY29sb3I6d2hpdGU7XG4gIH1cbiAgaDR7XG4gICAgZm9udC13ZWlnaHQ6IGJvbGQ7XG4gICAgbWFyZ2luLWJvdHRvbTogMC4yZW07XG4gIH1cblxuICAubWQtYnV0dG9uLm1kLXByaW1hcnkubWQtcmFpc2VkLCAubWQtYnV0dG9uLm1kLXByaW1hcnkubWQtZmFie1xuICAgIHBhZGRpbmc6IDZweCAyMHB4O1xuICAgIGhlaWdodDogNDRweDtcbiAgICBib3JkZXItcmFkaXVzOiAyMnB4O1xuICAgIGZvbnQtc2l6ZTogMThweDtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmx1ZTtcbiAgICBjb2xvcjp3aGl0ZTtcblxuICAgICY6aG92ZXIsICY6Zm9jdXN7XG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjM0M5OEFGO1xuICAgIH1cblxuICAgICY6bm90KFtkaXNhYmxlZF0pe1xuICAgICAgYm94LXNoYWRvdzogbm9uZTtcbiAgICB9XG4gIH1cblxuICAubWQtYnV0dG9uLm1kLXByaW1hcnl7XG4gICAgY29sb3I6ICRibHVlO1xuICB9XG59XG5cbi5ub3QtZm91bmQtNDA0IHtcbiAgaGVpZ2h0OiAzNnZoO1xufVxuXG5AbWVkaWEgKG1pbi13aWR0aDogOTYwcHgpIHtcbiAgLm5vdC1mb3VuZC00MDQge1xuICAgIGhlaWdodDogNjcuM3ZoO1xuICB9XG59XG5cbi8vIEFib3V0XG5cbi5zZW5zb3ItaW1hZ2UtbWFyZ2luIHtcbiAgbWFyZ2luLXRvcDogNDBweDtcbn1cblxuaDMubm8tbWFyZ2luIHtcbiAgbWFyZ2luOiAwO1xufVxuXG5wLm5vLW1hcmdpbiB7XG4gIG1hcmdpbjogMDtcbn1cblxuLnRlYW0tY2VsbHMtbWFyZ2luIHtcbiAgbWFyZ2luLWJvdHRvbTogMjVweDtcbn1cblxuLnN1YnRpdGxlLXNlcGFyYXRpb24ge1xuICBtYXJnaW4tdG9wOiA2MHB4O1xufVxuXG4vLyBGZWVkYmFja1xuXG4uZG9vcmJlbGwtYnV0dG9uIHtcbiBiYWNrZ3JvdW5kLWNvbG9yOiAkeWVsbG93ICFpbXBvcnRhbnQ7XG59XG5cbi8vIFN0eWxlZ3VpZGVcblxuLnN0eWxlZ3VpZGUge1xuICAuaW5mby10ZXh0IHtcbiAgICBjb2xvcjogcmdiYSgwLCAwLCAwLCAwLjMpO1xuICB9XG5cbiAgLmluZm8tdGV4dC1kYXJrIHtcbiAgICBjb2xvcjogcmdiYSgyNTUsIDI1NSwgMjU1LCAwLjMpICFpbXBvcnRhbnQ7XG4gIH1cblxuICAuZGFyay10aGVtZS1kaXZpZGVyIHtcbiAgICBib3JkZXItdG9wLWNvbG9yOiByZ2JhKDI1NSwyNTUsMjU1LDAuMTIpO1xuICB9XG5cbiAgc21hbGwge1xuICAgIG1hcmdpbi10b3A6IDIwcHg7XG4gIH1cblxuICAuZGFyay10ZXh0LXNlY3Rpb24ge1xuICAgIGJhY2tncm91bmQtY29sb3I6ICRiYWNrZ3JvdW5kX2JsdWVfZGFya19iYWNrZ3JvdW5kO1xuXG4gICAgaDEge1xuICAgICAgZm9udC1zaXplOiAxLjhlbTtcbiAgICAgIGNvbG9yOiAjZmZmO1xuICAgIH1cblxuICAgIGgyIHtcbiAgICAgIGNvbG9yOiAjZmZmO1xuICAgIH1cblxuICAgIGg0IHtcbiAgICAgIGNvbG9yOiAkZm9udF9jb2xvcl9ibHVlX2dyZXk7XG4gICAgfVxuXG4gICAgaDYge1xuICAgICAgY29sb3I6ICM4N0NDREQ7XG4gICAgfVxuXG4gICAgcHtcbiAgICAgIGNvbG9yOiAjQzhFNkVEO1xuICAgICAgZm9udC13ZWlnaHQ6IDMwMDtcbiAgICB9XG5cbiAgICBzbWFsbCB7XG4gICAgICBjb2xvcjogJHNlY29uZGFyeV9jb2xvcl9saWdodDtcbiAgICB9XG4gIH1cblxuICAuc2VjdGlvbi1wYWRkaW5nIHtcbiAgICBwYWRkaW5nOiA2MHB4O1xuICB9XG5cbiAgLmNvbG9ycy1zZWN0aW9uIHtcbiAgICBwIHtcbiAgICAgIHBhZGRpbmctbGVmdDogMjBweDtcbiAgICB9XG5cbiAgICAuc2Vjb25kYXJ5LWNvbG9yIHtcbiAgICAgIGJhY2tncm91bmQtY29sb3I6ICRzZWNvbmRhcnlfY29sb3I7XG4gICAgICBwIHtcbiAgICAgICAgY29sb3I6ICNmZmY7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLnRlcmNpYXJ5X2NvbG9yIHtcbiAgICAgIGJhY2tncm91bmQtY29sb3I6ICR0ZXJjaWFyeV9jb2xvcjtcbiAgICAgIHAge1xuICAgICAgICBjb2xvcjogI2ZmZjtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAuc2Vjb25kYXJ5X2NvbG9yX2xpZ2h0IHtcbiAgICAgIGJhY2tncm91bmQtY29sb3I6ICM4REIyQkE7XG4gICAgICBwIHtcbiAgICAgICAgY29sb3I6ICNmZmY7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLnNlY29uZGFyeS1jb2xvci1wYXN0ZWwge1xuICAgICAgYmFja2dyb3VuZC1jb2xvcjogI0M4RTZFRDtcbiAgICAgIHAge1xuICAgICAgICBjb2xvcjogIzhEQjJCQTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAud2hpdGUge1xuICAgICAgYmFja2dyb3VuZC1jb2xvcjogI2ZmZjtcbiAgICAgIGJvcmRlcjoxcHggc29saWQgI0M4RTZFRDtcbiAgICAgIHAge1xuICAgICAgICBjb2xvcjogIzhEQjJCQTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAuc2Vuc29ycyA+IGRpdiB7XG4gICAgICB3aWR0aDogNjBweDtcbiAgICAgIGhlaWdodDogNjBweDtcbiAgICAgIGJvcmRlci1yYWRpdXM6IDMwcHg7XG4gICAgICBmbG9hdDogbGVmdDtcbiAgICAgIG1hcmdpbjogMTBweDtcbiAgICB9XG5cbiAgICAuc2Vuc29ycyBkaXY6bnRoLWNoaWxkKDEpIHtcbiAgICAgYmFja2dyb3VuZDogI2ZmYzEwNztcbiAgICB9XG5cbiAgICAuc2Vuc29ycyBkaXY6bnRoLWNoaWxkKDIpIHtcbiAgICAgYmFja2dyb3VuZDogIzRmYzNmNztcbiAgICB9XG5cbiAgICAuc2Vuc29ycyBkaXY6bnRoLWNoaWxkKDMpIHtcbiAgICAgYmFja2dyb3VuZDogI2ZmZWU1ODtcbiAgICB9XG5cbiAgICAuc2Vuc29ycyBkaXY6bnRoLWNoaWxkKDQpIHtcbiAgICAgYmFja2dyb3VuZDogI2YwNjI5MjtcbiAgICB9XG5cbiAgICAuc2Vuc29ycyBkaXY6bnRoLWNoaWxkKDUpIHtcbiAgICAgYmFja2dyb3VuZDogIzRjYWY1MDtcbiAgICB9XG5cbiAgICAuc2Vuc29ycyBkaXY6bnRoLWNoaWxkKDYpIHtcbiAgICAgYmFja2dyb3VuZDogIzhiYzM0YTtcbiAgICB9XG5cbiAgICAuc2Vuc29ycyBkaXY6bnRoLWNoaWxkKDcpIHtcbiAgICAgYmFja2dyb3VuZDogIzk1NzVjZDtcbiAgICB9XG5cbiAgICAuc2Vuc29ycyBkaXY6bnRoLWNoaWxkKDgpIHtcbiAgICAgYmFja2dyb3VuZDogI2ZmZjljNDtcbiAgICB9XG5cbiAgICAuc2Vuc29ycyBkaXY6bnRoLWNoaWxkKDkpIHtcbiAgICAgYmFja2dyb3VuZDogI2ZmZWU1ODtcbiAgICB9XG5cbiAgICAuc2Vuc29ycyBkaXY6bnRoLWNoaWxkKDEwKSB7XG4gICAgIGJhY2tncm91bmQ6ICNmZjU3MjI7XG4gICAgfVxuICB9XG5cbn1cblxuXG4vKiBDdXN0b20gQW5pbWF0aW9ucyAqL1xuXG4ubG9hZGluZy1kb3RzIHNwYW46YWZ0ZXIge1xuICBjb250ZW50OiAnLic7XG4gIGFuaW1hdGlvbjogZG90cyAycyBzdGVwcyg1LCBlbmQpIGluZmluaXRlO31cblxuQGtleWZyYW1lcyBkb3RzIHtcbiAgMCUsIDIwJSB7XG4gICAgY29sb3I6IHJnYmEoMCwwLDAsMCk7XG4gICAgdGV4dC1zaGFkb3c6XG4gICAgICAuMjVlbSAwIDAgcmdiYSgwLDAsMCwwKSxcbiAgICAgIC41ZW0gMCAwIHJnYmEoMCwwLDAsMCk7fVxuICA0MCUge1xuICAgIGNvbG9yOiB3aGl0ZTtcbiAgICB0ZXh0LXNoYWRvdzpcbiAgICAgIC4yNWVtIDAgMCByZ2JhKDAsMCwwLDApLFxuICAgICAgLjVlbSAwIDAgcmdiYSgwLDAsMCwwKTt9XG4gIDYwJSB7XG4gICAgdGV4dC1zaGFkb3c6XG4gICAgICAuMjVlbSAwIDAgJHllbGxvdyxcbiAgICAgIC41ZW0gMCAwIHJnYmEoMCwwLDAsMCk7fVxuICA4MCUsIDEwMCUge1xuICAgIHRleHQtc2hhZG93OlxuICAgICAgLjI1ZW0gMCAwICR5ZWxsb3csXG4gICAgICAuNWVtIDAgMCAkYmx1ZTt9fVxuXG5cbi8qIERvIG5vdCByZW1vdmUgdGhpcyBjb21tZW50cyBiZWxsb3cuIEl0J3MgdGhlIG1hcmtlcnMgdXNlZCBieSBndWxwLWluamVjdCB0byBpbmplY3RcbiAgIGFsbCB5b3VyIHNhc3MgZmlsZXMgYXV0b21hdGljYWxseSAqL1xuLy8gaW5qZWN0b3Jcbi8vIGVuZGluamVjdG9yXG4iLCIvLyBHZW5lcmFsIGhlbHBlcnNcbi5ib3JkZXItYmxhY2t7XG4gIGJvcmRlcjogM3B4IHNvbGlkICRvZmZfYmxhY2s7XG59XG4uYm9yZGVyLXdoaXRle1xuICBib3JkZXI6IDNweCBzb2xpZCB3aGl0ZTtcbn1cbi5jb2xvci13aGl0ZXtcbiAgY29sb3I6IHdoaXRlICFpbXBvcnRhbnQ7XG59XG4uY29sb3ItZHJvcGRvd257XG4gIGNvbG9yOiAkZ3JleV9kYXJrZXN0ICFpbXBvcnRhbnQ7XG59XG4uY29sb3ItYmxhY2t7XG4gIGNvbG9yOiAkb2ZmX2JsYWNrO1xufVxuLmNvbG9yLWJsdWV7XG4gIGNvbG9yOiR0ZXJjaWFyeV9jb2xvcjtcbn1cbi5jb2xvci1jeWFue1xuICBjb2xvcjogJHRlcmNpYXJ5X2NvbG9yO1xufVxuLmNvbG9yLWdyZWVuIHtcbiAgY29sb3I6ICRncmVlbjtcbiAgc3ZnIHtcbiAgICBmaWxsOiAkZ3JlZW47XG4gIH1cbn1cbi5jb2xvci1yZWQge1xuICBjb2xvcjogJHJlZDtcbiAgc3ZnIHtcbiAgICBmaWxsOiAkcmVkICFpbXBvcnRhbnQ7XG4gIH1cbn1cbi5mbG9hdC1sZWZ0e1xuICBmbG9hdDogbGVmdDtcbn1cbi5mbG9hdC1yaWdodHtcbiAgZmxvYXQ6IHJpZ2h0O1xufVxuLmQtZmxleHtcbiAgZGlzcGxheTpmbGV4O1xufVxuLmZ1bGwtd2lkdGh7XG4gIHdpZHRoOiAxMDAlO1xufVxuLm1heC13aWR0aC01MDBweCB7XG4gIG1heC13aWR0aDogNTAwcHg7XG59XG4ubWluLWhlaWdodC04MCB7XG4gIG1pbi1oZWlnaHQ6IDgwJTtcbn1cblxuLy8gRm9udCAmIHRleHRcbi5mb250LXJvYm90by1jb25kZW5zZWR7XG4gIGZvbnQtZmFtaWx5OidSb2JvdG8gQ29uZGVuc2VkJztcbn1cbi5mb250LWthbml0e1xuICBmb250LWZhbWlseTonS2FuaXQnO1xufVxuXG4udGV4dC1jZW50ZXJ7XG4gIHRleHQtYWxpZ246IGNlbnRlcjtcbn1cbi50ZXh0LWxlZnR7XG4gIHRleHQtYWxpZ246IGxlZnQ7XG59XG5cblxuXG4vLyBCYWNrZ3JvdW5kIGhlbHBlcnNcbi5iZy13aGl0ZXtcbiAgYmFja2dyb3VuZC1jb2xvcjogI2ZmZjtcbn1cbi5iZy1ibGFja3tcbiAgYmFja2dyb3VuZC1jb2xvcjogYmxhY2s7XG59XG4uYmctZ3JlZW57XG4gIGJhY2tncm91bmQtY29sb3I6ICRncmVlbjtcbn1cbi5iZy1ibHVle1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmx1ZTtcbn1cbi5iZy1yZWR7XG4gIGJhY2tncm91bmQtY29sb3I6ICRyZWQgIWltcG9ydGFudDtcbn1cbi5iZy1yZWQtbGlnaHR7XG4gIGJhY2tncm91bmQtY29sb3I6IHJnYmEoJHJlZCwuMik7XG59XG4uYmcteWVsbG93e1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeWVsbG93O1xufVxuLmJnLWdyZXkge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkZ3JleSAhaW1wb3J0YW50O1xufVxuLmJnLWdyZXktbGlnaHRlc3Qge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkZ3JleV9saWdodGVzdCAhaW1wb3J0YW50O1xufVxuXG4ub3V0bGluZXtcbiAgb3V0bGluZTogMXB4IHNvbGlkIHJlZDtcbn1cblxuLy8gQnV0dG9uIGhlbHBlcnNcbi5idG4tb3V0bGluZS1ibHVle1xuICBib3JkZXI6MnB4IHNvbGlkICRibHVlO1xuICBjb2xvcjogJGJsdWU7XG4gIHBhZGRpbmc6MTBweCAyOXB4O1xuICBzdmcge1xuICAgIGZpbGw6ICRibHVlXG4gIH1cbn1cbi5idG4tb3V0bGluZS1ibHVlOmhvdmVye1xuICBiYWNrZ3JvdW5kOiAkYmx1ZSAhaW1wb3J0YW50O1xuICBjb2xvcjogd2hpdGU7XG4gIHN2ZyB7XG4gICAgZmlsbDogd2hpdGVcbiAgfVxufVxuLmJ0bi1vdXRsaW5lLXdoaXRle1xuICBib3JkZXI6MnB4IHNvbGlkIHdoaXRlO1xuICBjb2xvcjogd2hpdGU7XG4gIHBhZGRpbmc6MTBweCAyOXB4O1xufVxuLmJ0bi1vdXRsaW5lLXdoaXRlOmhvdmVye1xuICBiYWNrZ3JvdW5kOiB3aGl0ZSAhaW1wb3J0YW50O1xuICBjb2xvcjogJG9mZl9ibGFjaztcbn1cbi5idG4tb3V0bGluZS13aGl0ZS1ibHVle1xuICBib3JkZXI6MnB4IHNvbGlkIHdoaXRlO1xuICBjb2xvcjogd2hpdGU7XG4gIHBhZGRpbmc6MTBweCAyOXB4O1xufVxuLmJ0bi1vdXRsaW5lLXdoaXRlLWJsdWU6aG92ZXJ7XG4gIGJhY2tncm91bmQ6ICRibHVlICFpbXBvcnRhbnQ7XG59XG4uYnRuLW91dGxpbmUteWVsbG93e1xuICBib3JkZXI6MnB4IHNvbGlkICR5ZWxsb3c7XG4gIGNvbG9yOiAkeWVsbG93O1xuICBwYWRkaW5nOjEwcHggMjlweDtcbn1cbi5idG4tb3V0bGluZS15ZWxsb3c6aG92ZXJ7XG4gIGJhY2tncm91bmQ6ICR5ZWxsb3cgIWltcG9ydGFudDtcbiAgY29sb3I6ICRvZmZfYmxhY2s7XG59XG4uYnRuLXllbGxvd3tcbiAgYm9yZGVyOjJweCBzb2xpZCAkeWVsbG93O1xuICBiYWNrZ3JvdW5kOiAkeWVsbG93O1xuICBjb2xvcjogJG9mZl9ibGFjaztcbiAgcGFkZGluZzoxMnB4IDI5cHg7XG59XG5cbi5idG4teWVsbG93OmhvdmVye1xuICBiYWNrZ3JvdW5kOiAkb2ZmX2JsYWNrICFpbXBvcnRhbnQ7XG4gIGNvbG9yOiAkeWVsbG93O1xufVxuLmJ0bi1ibGFjay1vdXRsaW5le1xuICBib3JkZXI6MnB4IHNvbGlkICRvZmZfYmxhY2s7XG4gIGNvbG9yOiAkb2ZmX2JsYWNrO1xuICBwYWRkaW5nOjEwcHggMjlweDtcbn1cbi5idG4tYmxhY2stb3V0bGluZTpob3ZlcntcbiAgYmFja2dyb3VuZC1jb2xvcjogJG9mZl9ibGFjayAhaW1wb3J0YW50O1xuICBjb2xvcjogd2hpdGU7XG59XG4uYnRuLWJsdWV7XG4gIGJvcmRlcjoycHggc29saWQgJGJsdWU7XG4gIGJhY2tncm91bmQ6ICRibHVlO1xuICBjb2xvcjogd2hpdGU7XG4gIHBhZGRpbmc6MTJweCAyOXB4O1xufVxuXG4uYnRuLWJsdWU6aG92ZXJ7XG4gIGNvbG9yOiAkYmx1ZTtcbiAgYmFja2dyb3VuZDogJGdyZXlfbGlnaHRlciAhaW1wb3J0YW50O1xufVxuLmJ0bi13aGl0ZS1ibHVle1xuICBib3JkZXI6MnB4IHNvbGlkIHdoaXRlO1xuICBiYWNrZ3JvdW5kOiBub25lO1xuICBjb2xvcjogd2hpdGU7XG4gIHBhZGRpbmc6MTJweCAyOXB4O1xufVxuXG4uYnRuLXdoaXRlLWJsdWU6aG92ZXJ7XG4gIGNvbG9yOiAkYmx1ZTtcbiAgYmFja2dyb3VuZDogd2hpdGUgIWltcG9ydGFudDtcbiAgYm9yZGVyOjJweCBzb2xpZCAkYmx1ZTtcbn1cblxuLy8gVE9ETzogQ29zbWV0aWMgRml4IGJ1dHRvbnMgaW4gZ2VuZXJhbFxuLmJ0bi1yb3VuZC1uZXcge1xuICBib3JkZXItcmFkaXVzOiA0MHB4O1xuICBmb250LWZhbWlseTogJ1JvYm90byBDb25kZW5zZWQnO1xuICBmb250LXNpemU6IDE2cHg7XG4gIC8vIGhlaWdodDogNDRweDtcbiAgbGluZS1oZWlnaHQ6IDQwcHg7XG4gIGZvbnQtd2VpZ2h0OiAkZm9udC13ZWlnaHQtaGVhdnk7XG4gIGxldHRlci1zcGFjaW5nOiAwLjAzMjVlbTtcbiAgcGFkZGluZzowcHggMzVweDtcbiAgdHJhbnNpdGlvbjogYm94LXNoYWRvdyAwLjRzIGN1YmljLWJlemllcigwLjI1LCAwLjgsIDAuMjUsIDEpLCBiYWNrZ3JvdW5kLWNvbG9yIDAuNHMgY3ViaWMtYmV6aWVyKDAuMjUsIDAuOCwgMC4yNSwgMSk7XG59XG5cbi5idG4tcm91bmR7XG4gIGJvcmRlcjogMDtcbiAgYm9yZGVyLXJhZGl1czogMjVweDtcbiAgcGFkZGluZzo1cHggMzVweDtcbiAgZm9udC1zaXplOjE4cHg7XG59XG5cbi5idG4tY3lhbntcbiAgYmFja2dyb3VuZC1jb2xvcjogJHRlcmNpYXJ5X2NvbG9yO1xuICBjb2xvcjp3aGl0ZTtcbiAgYm9yZGVyOjJweCBzb2xpZCAkdGVyY2lhcnlfY29sb3I7XG59XG5cbi5idG4tY3lhbjpob3ZlcntcbiAgY29sb3I6ICR0ZXJjaWFyeV9jb2xvcjtcbiAgYmFja2dyb3VuZDogd2hpdGUgIWltcG9ydGFudDtcbiAgYm9yZGVyOjJweCBzb2xpZCAkdGVyY2lhcnlfY29sb3I7XG59XG5cbi8vIGRpc3BsYXkgaGVscGVyc1xuLmJ0bi1mdWxsIHtcbiAgZGlzcGxheTogYmxvY2s7XG4gIHdpZHRoOiAxMDAlO1xuICBib3JkZXItcmFkaXVzOiAwcHg7XG4gIG1hcmdpbjogMHB4O1xuICBwYWRkaW5nOjEycHggMjlweDtcbn1cblxuLmJ0bi1zbWFsbCB7XG4gIG1pbi13aWR0aDogdW5zZXQgIWltcG9ydGFudDtcbiAgbWluLWhlaWdodDogdW5zZXQgIWltcG9ydGFudDtcbiAgLy9vdXRsaW5lOiAxcHggc29saWQgcmVkO1xufVxuXG4vLyBNYXJnaW4gLSBwYWRkaW5nIGhlbHBlcnNcbi5tLTB7XG4gIG1hcmdpbjogMDtcbn1cbi5tLTEwe1xuICBtYXJnaW46IDEwcHg7XG59XG4ubWwtMCB7XG4gIG1hcmdpbi1sZWZ0OiAwcHg7XG59XG4ubWwtMTUge1xuICBtYXJnaW4tbGVmdDogMTVweDtcbn1cbi5tdC0xMHsgbWFyZ2luLXRvcDogMTBweDsgfVxuLm10LTIweyBtYXJnaW4tdG9wOiAyMHB4OyB9XG4ubXQtMzB7IG1hcmdpbi10b3A6IDMwcHg7IH1cbi5tdC01MHsgbWFyZ2luLXRvcDogNTBweDsgfVxuLm1yLTEweyBtYXJnaW4tcmlnaHQ6IDEwcHg7IH1cbi5tci0yMHsgbWFyZ2luLXJpZ2h0OiAyMHB4OyB9XG4ubXItMzB7IG1hcmdpbi1yaWdodDogMzBweDsgfVxuLm1yLTUweyBtYXJnaW4tcmlnaHQ6IDUwcHg7IH1cbi5tYi0xMHsgbWFyZ2luLWJvdHRvbTogMTBweDsgfVxuLm1iLTIweyBtYXJnaW4tYm90dG9tOiAyMHB4OyB9XG4ubWItMzB7IG1hcmdpbi1ib3R0b206IDMwcHg7IH1cbi5tYi01MHsgbWFyZ2luLWJvdHRvbTogNTBweDsgfVxuLm1sLTEwIHsgbWFyZ2luLWxlZnQ6IDEwcHg7IH1cbi5tbC0yMCB7IG1hcmdpbi1sZWZ0OiAyMHB4OyB9XG4ubWwtMzAgeyBtYXJnaW4tbGVmdDogMzBweDsgfVxuLm1sLTUweyBtYXJnaW4tbGVmdDogNTBweDsgfVxuLm15LTIwe1xuICBtYXJnaW4tdG9wOiAyMHB4ICFpbXBvcnRhbnQ7XG4gIG1hcmdpbi1ib3R0b206IDIwcHggIWltcG9ydGFudDtcbn1cbi5tdy0xMDB7XG4gIG1heC13aWR0aDogMTAwJTtcbn1cbi5wLTB7IHBhZGRpbmc6IDA7IH1cbi5wLTEweyBwYWRkaW5nOiAxMHB4OyB9XG4ucC0yMHsgcGFkZGluZzogMjBweDsgfVxuLnAtMzB7IHBhZGRpbmc6IDMwcHg7IH1cbi5wLTQweyBwYWRkaW5nOiA0MHB4OyB9XG4ucC01MHsgcGFkZGluZzogNTBweDsgfVxuLnAtNjB7IHBhZGRpbmc6IDYwcHg7IH1cbi5wbC0yMHsgcGFkZGluZy1sZWZ0OiAyMHB4OyB9XG4ucHQtODB7IHBhZGRpbmctdG9wOjgwcHg7IH1cbi5weS00MHtcbiAgcGFkZGluZy10b3A6NDBweDtcbiAgcGFkZGluZy1ib3R0b206NDBweDtcbn1cbi5weC0yMHtcbiAgcGFkZGluZy1sZWZ0OjIwcHg7XG4gIHBhZGRpbmctcmlnaHQ6MjBweDtcbn1cbi5weC00MHtcbiAgcGFkZGluZy1sZWZ0OjQwcHg7XG4gIHBhZGRpbmctcmlnaHQ6NDBweDtcbn1cblxuXG4vLyBpbWcgaGVscGVyc1xuXG4uaW1nLWNpcmNsZSB7XG4gIGJvcmRlci1yYWRpdXM6IDUwJTtcbn1cblxuZm9vdGVye1xuICBwe1xuICAgIGZvbnQtc2l6ZToxMnB4ICFpbXBvcnRhbnQ7XG4gICAgbGluZS1oZWlnaHQ6MTJweCAhaW1wb3J0YW50O1xuICB9XG59XG5cbi8vIHBvc3Rpb24gaGVscGVyc1xuXG4uYWJzb2x1dGUge1xuICBwb3NpdGlvbjogYWJzb2x1dGU7XG59XG5cbi5yZWxhdGl2ZSB7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbn1cblxuLy8gc2l6ZSBoZWxwZXJzXG4ucy00OCB7XG4gIHdpZHRoOiA0OHB4O1xuICBoZWlnaHQ6IDQ4cHg7XG59XG4iLCIkeWVsbG93OiAjRkZDMTAwO1xuJHllbGxvd19saWdodDogI0ZGRDQ0RDtcbiR5ZWxsb3dfbGlnaHRlcjogI0ZGRUNCMztcbiRibHVlOiAjMDAxOUZGO1xuJGJsdWVfbGlnaHQ6ICM1NUM0RjU7XG5cbiRyZWQ6ICNGRjNENEM7XG4kZ3JlZW46ICMwMEU1OTc7XG4vLyAkb2ZmX2JsYWNrOiAjMkUyRTJFO1xuLy8gJG9mZl9kYXJrZXJfYmxhY2s6ICMxQzFDMUM7XG4kb2ZmX2xpZ2h0ZXJfYmxhY2s6ICMyRTJFMkU7XG4kb2ZmX2JsYWNrOiAjMUMxQzFDO1xuXG4kZ3JleV9saWdodGVzdDogI0Y5RjlGQjtcbiRncmV5X2xpZ2h0ZXI6ICNFQ0VDRUU7XG4kZ3JleTogI0QyRDNENTtcbiRncmV5X2RhcmtlcjogIzZFNkU2RTtcbiRncmV5X2Rhcmtlc3Q6ICM3RTdFN0U7XG5cbiRzZWNvbmRhcnlfY29sb3I6ICRncmV5X2RhcmtlcjtcbiR0ZXJjaWFyeV9jb2xvcjogJGJsdWU7XG5cbiRzZWNvbmRhcnlfY29sb3JfbGlnaHQ6ICRncmV5X2RhcmtlcjtcbiR0ZXJjaWFyeV9jb2xvcl9saWdodDogJGdyZXk7XG5cbiRpbmZvX3ByaW1hcnk6ICRncmV5O1xuJGluZm9fc2Vjb25kYXJ5OiAkc2Vjb25kYXJ5X2NvbG9yX2xpZ2h0O1xuXG4kYWxlcnRfcmVkOiAkcmVkO1xuJGFsZXJ0X2dyZWVuOiAjNERCRDRFO1xuXG4kZm9udF9jb2xvcl93aGl0ZTogd2hpdGU7XG4kZm9udF9jb2xvcl9ibHVlX2dyZXk6ICRncmV5O1xuJGZvbnRfY29sb3JfYmx1ZV9kYXJrOiAkZ3JleV9kYXJrZXI7XG4kZm9udF9jb2xvcl9ibHVlX2xpZ2h0OiAkYmx1ZTtcblxuJGJhY2tncm91bmRfYmx1ZV9kYXJrX21lbnU6ICRvZmZfYmxhY2s7XG4kYmFja2dyb3VuZF9ibHVlX2xpZ2h0ZXJfbWVudTogJG9mZl9saWdodGVyX2JsYWNrO1xuJGJhY2tncm91bmRfYmx1ZV9kYXJrX292ZXJ2aWV3OiAkb2ZmX2JsYWNrO1xuJGJhY2tncm91bmRfYmx1ZV9kYXJrX2JhY2tncm91bmQ6ICRvZmZfYmxhY2s7XG4kYmFja2dyb3VuZF9ibHVlX2RhcmtfbmF2YmFyOiAkb2ZmX2JsYWNrO1xuJGJhY2tncm91bmRfYmx1ZV9kYXJrX2NoYXJ0OiAkb2ZmX2JsYWNrO1xuJGJhY2tncm91bmRfYmx1ZV9saWdodDogJGdyZXlfbGlnaHRlcjtcbiRiYWNrZ3JvdW5kX2JsdWVfbWlkdG9uZTogJGdyZXk7XG5cbiRsYXJnZV9wcmlvcml0eTogMTA7XG4kbW9yZV9wcmlvcml0eTogNTtcbiRtZWRpdW1fcHJpb3JpdHk6IDI7XG4kc21hbGxfcHJpb3JpdHk6IC0xO1xuXG4kZHJvcGRvd25fYmx1ZTogIzNmNTFiNTtcblxuJGZvbnQtd2VpZ2h0LWhlYXZ5OiA2MDA7XG4kdGltZWxpbmUtaGVpZ2h0OiAxMjBweDtcbiRjb250ZW50LW1hcmdpbi10b3A6IDIyMHB4O1xuIiwiLy8gTmV3IGxhbmRpbmcgcGFnZVxuXG4uZ3JleS13YXZlc3tcbiAgYmFja2dyb3VuZDogZ3JleSB1cmwoXCIvYXNzZXRzL2ltYWdlcy9zY2tfYmcucG5nXCIpIDtcbiAgYmFja2dyb3VuZC1yZXBlYXQ6IHJlcGVhdDtcbiAgYmFja2dyb3VuZC1zaXplOiA2NXB4O1xufVxuXG4ubmV3LWxhbmRpbmctcGFnZXtcbiAgaDEsaDIsaDMsaDQsaDUsaDZ7XG4gICAgbGV0dGVyLXNwYWNpbmc6IDAuMDMyNWVtO1xuICAgIGZvbnQtZmFtaWx5OiAnS2FuaXQnO1xuICAgIGZvbnQtd2VpZ2h0OiA5MDA7XG4gICAgY29sb3I6ICRvZmZfYmxhY2s7XG4gIH1cbiAgaDF7XG4gICAgZm9udC1zaXplOiA2NXB4O1xuICAgIGxpbmUtaGVpZ2h0OiA2NXB4O1xuICAgIG1hcmdpbjowcHggMHB4O1xuICAgIHBhZGRpbmctdG9wOiA3MHB4O1xuICB9XG4gIGgye1xuICAgIGZvbnQtc2l6ZTogNDVweDtcbiAgICBsaW5lLWhlaWdodDogNDVweDtcbiAgICBtYXJnaW46MHB4IDBweDtcbiAgfVxuICBoM3tcbiAgICBmb250LXNpemU6IDMwcHg7XG4gICAgbGluZS1oZWlnaHQ6IDMwcHg7XG4gICAgbWFyZ2luLXRvcDoxMHB4O1xuICAgIG1hcmdpbi1ib3R0b206IDEwcHg7XG4gIH1cbiAgcHtcbiAgICBjb2xvcjogJG9mZl9ibGFjaztcbiAgICBmb250LWZhbWlseTogJ1JvYm90byBDb25kZW5zZWQnO1xuICAgIGZvbnQtc2l6ZTogMjBweDtcbiAgICBsaW5lLWhlaWdodDogMjRweDtcbiAgfVxuICBhe1xuICAgIHRleHQtZGVjb3JhdGlvbjogbm9uZSAhaW1wb3J0YW50O1xuICB9XG4gIC5zYy1sb2dve1xuICAgIGhlaWdodDogNDNweDtcbiAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgei1pbmRleDogMjtcbiAgICBsZWZ0OiAxMCU7XG4gICAgdG9wOjI1cHg7XG4gIH1cbiAgLnNjLW9mZi1jdGEtcGxhdGZvcm0ge1xuICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICB6LWluZGV4OiAyO1xuICAgIHJpZ2h0OiAxMCU7XG4gICAgdG9wOjI1cHg7XG5cbiAgfVxuICAvLyBJbWFnZXMgb24gbGFuZGluZyBwYWdlXG4gIC5pbWctbmV3X3Nja3tcbiAgICBtaW4taGVpZ2h0OiAzMHZ3O1xuICAgIGJhY2tncm91bmQ6IGJsdWUgdXJsKFwiL2Fzc2V0cy9pbWFnZXMvbmV3X3Njay5qcGdcIikgY2VudGVyL2NvdmVyO1xuICB9XG4gIC5pbWctc2NrX2VkdXtcbiAgICBtaW4taGVpZ2h0OiAzMHZ3O1xuICAgIGJhY2tncm91bmQ6IGJsdWUgdXJsKFwiL2Fzc2V0cy9pbWFnZXMvc2NrX2VkdS5qcGdcIikgY2VudGVyL2NvdmVyO1xuICB9XG4gIC5pbWctc2NrX2NvbXtcbiAgICBiYWNrZ3JvdW5kOiBibHVlIHVybChcIi9hc3NldHMvaW1hZ2VzL3Nja19jb21tdW5pdGllcy5qcGdcIikgY2VudGVyL2NvdmVyO1xuICB9XG4gIC5pbWctcmVzZWFyY2h7XG4gICAgYmFja2dyb3VuZDogYmx1ZSB1cmwoXCIvYXNzZXRzL2ltYWdlcy9zY2tfcmVzZWFyY2guanBnXCIpIGNlbnRlci9jb3ZlcjtcbiAgfVxuICAuaW1nLWdvdmVybm17XG4gICAgYmFja2dyb3VuZDogYmx1ZSB1cmwoXCIvYXNzZXRzL2ltYWdlcy9zY2tfY2l0aWVzLmpwZ1wiKSBjZW50ZXIvY292ZXI7XG4gIH1cbiAgLmltZy1wbGF0Zm9ybXtcbiAgICBtaW4taGVpZ2h0OjYzMHB4ICFpbXBvcnRhbnQ7XG4gICAgYmFja2dyb3VuZDogYmx1ZSB1cmwoXCIvYXNzZXRzL2ltYWdlcy9zY2tfcGxhdGZvcm0uanBnXCIpIGNlbnRlci9jb3ZlcjtcbiAgfVxuICAuaW1nLWRvY3N7XG4gICAgbWluLWhlaWdodDozMjBweCAhaW1wb3J0YW50O1xuICAgIGJhY2tncm91bmQ6IGJsdWUgdXJsKFwiL2Fzc2V0cy9pbWFnZXMvc2NrX2RvY3MuanBnXCIpIHRvcC9jb3ZlcjtcbiAgfVxuICAvLyBUaWxlcyBvbiBsYW5kaW5nIHBhZ2VcbiAgLnRpbGV7XG4gICAgcGFkZGluZzogNjBweCA0MHB4O1xuICAgIGJvcmRlci1ib3R0b206IDNweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICAgIGJvcmRlci1yaWdodDogM3B4IHNvbGlkICRvZmZfYmxhY2s7XG4gIH1cbiAgLnRpbGUtaW1hZ2V7XG4gICAgbWluLWhlaWdodDozMDBweDtcbiAgICBwYWRkaW5nOjAgIWltcG9ydGFudDtcbiAgfVxuXG4gIC50aWxlLWxlZnR7XG4gICAgYm9yZGVyLWxlZnQ6IDNweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICB9XG4gIC50aWxlLXRvcHtcbiAgICBib3JkZXItdG9wOiAzcHggc29saWQgJG9mZl9ibGFjaztcbiAgfVxuXG4gIC50ZXh0LWZ1bmRpbmd7XG4gICAgdGV4dC1hbGlnbjogbGVmdDtcbiAgfVxuXG4gIC8vIFZpZGVvIHNlY3Rpb25cblxuICAudmlkZW8tc2VjdGlvbntcbiAgICBiYWNrZ3JvdW5kOiBibHVlIHVybChcIi9hc3NldHMvaW1hZ2VzL2xhbmRpbmcvc21hcnRjaXRpemVuLXNlZWVkLXByZW9yZGVyLmpwZ1wiKSBjZW50ZXIvY292ZXI7XG4gICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgIGhlaWdodDogOTB2aDtcbiAgICB3aWR0aDogMTAwJTtcbiAgICBvdmVyZmxvdzogaGlkZGVuO1xuICB9XG4gIC5oZWFkaW5nLW92ZXItdmlkZW97XG4gICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgIHotaW5kZXg6IDE7XG4gICAgdG9wOiAwcHg7XG4gICAgaGVpZ2h0OiAxMDAlO1xuICAgIHBhZGRpbmctbGVmdDogMTAlO1xuICAgIHBhZGRpbmctcmlnaHQ6IDYwJTtcblxuICAgIGltZ3tcbiAgICAgIHBhZGRpbmctYm90dG9tOjQwcHg7XG4gICAgfVxuICB9XG5cbiAgQG1lZGlhIG9ubHkgc2NyZWVuIGFuZCAobWF4LXdpZHRoOiAxMDEwcHgpIHtcbiAgICBoMXtcbiAgICAgIGZvbnQtc2l6ZTo0MnB4O1xuICAgICAgbGluZS1oZWlnaHQ6NDJweDtcbiAgICB9XG4gICAgaDJ7XG4gICAgICBmb250LXNpemU6MjhweDtcbiAgICAgIGxpbmUtaGVpZ2h0OjI4cHg7XG4gICAgfVxuICAgIGgze1xuICAgICAgZm9udC1zaXplOiAyNnB4O1xuICAgICAgbGluZS1oZWlnaHQ6IDI2cHg7XG4gICAgfVxuICAgIHB7XG4gICAgICBmb250LXNpemU6MThweDtcbiAgICAgIGxpbmUtaGVpZ2h0OjIycHg7XG4gICAgfVxuICAgIC5zYy1sb2dve1xuICAgICAgbGVmdDo1JTtcbiAgICB9XG4gICAgLmhlYWRpbmctb3Zlci12aWRlb3tcbiAgICAgIHBhZGRpbmctdG9wOiA0MHB4O1xuICAgICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgICAgcGFkZGluZy1sZWZ0OiA1JTtcbiAgICAgIHBhZGRpbmctcmlnaHQ6MjUlO1xuICAgICAgaGVpZ2h0OiAxMDAlO1xuICAgICAgaW1ne1xuICAgICAgICBwYWRkaW5nLWJvdHRvbTowMHB4O1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIC8vIExlc3MgdGhlbiA3NTBweFxuICBAbWVkaWEgb25seSBzY3JlZW4gYW5kIChtYXgtd2lkdGg6IDc1MHB4KSB7XG4gICAgaDF7XG4gICAgICBmb250LXNpemU6MzBweDtcbiAgICB9XG4gICAgaDJ7XG4gICAgICBmb250LXNpemU6MjVweDtcbiAgICB9XG4gICAgLnRleHQtZnVuZGluZ3tcbiAgICAgIHRleHQtYWxpZ246IGNlbnRlcjtcbiAgICB9XG4gIH1cblxuICBAbWVkaWEgb25seSBzY3JlZW4gYW5kIChtYXgtd2lkdGg6IDYwMHB4KSB7XG4gICAgaDF7XG4gICAgICBmb250LXNpemU6MjhweDtcbiAgICAgIGxpbmUtaGVpZ2h0OjMzcHg7XG4gICAgICBwYWRkaW5nLWJvdHRvbToxMHB4O1xuICAgIH1cbiAgICBoMntcbiAgICAgIGZvbnQtc2l6ZToyOHB4O1xuICAgIH1cbiAgICBwe1xuICAgICAgZm9udC1zaXplOjE5cHg7XG4gICAgfVxuICAgIC5wLTYwe1xuICAgICAgcGFkZGluZzogNTBweCAxMHB4ICFpbXBvcnRhbnQ7XG4gICAgfVxuICAgIC5ib3JkZXIteHMtdG9we1xuICAgICAgYm9yZGVyLXRvcDogM3B4IHNvbGlkICRvZmZfYmxhY2s7XG4gICAgICBib3JkZXItbGVmdDogM3B4IHNvbGlkICRvZmZfYmxhY2s7XG4gICAgICBib3JkZXItcmlnaHQ6IDNweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICAgICAgYm9yZGVyLWJvdHRvbTogM3B4IHNvbGlkICRvZmZfYmxhY2s7XG4gICAgfVxuICAgIC5ib3JkZXIteHMtYm90dG9te1xuICAgICAgYm9yZGVyLXRvcDogdW5zZXQ7XG4gICAgICBib3JkZXItbGVmdDogM3B4IHNvbGlkICRvZmZfYmxhY2s7XG4gICAgICBib3JkZXItcmlnaHQ6IDNweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICAgICAgYm9yZGVyLWJvdHRvbTozcHggc29saWQgJG9mZl9ibGFjaztcbiAgICB9XG4gICAgLmltZy1wbGF0Zm9ybXtcbiAgICAgIG1pbi1oZWlnaHQ6MzU2cHggIWltcG9ydGFudDtcbiAgICB9XG4gICAgLmhlYWRpbmctb3Zlci12aWRlb3tcbiAgICAgIHBhZGRpbmctcmlnaHQ6IDUlO1xuICAgIH1cbiAgICAuc2Mtb2ZmLWN0YS1wbGF0Zm9ybSB7XG4gICAgICBwYWRkaW5nOiAxMHB4IDEwcHg7XG4gICAgfVxuICAgIC5teS0yMHtcbiAgICAgIG1hcmdpbi10b3A6IDEwcHggIWltcG9ydGFudDtcbiAgICAgIG1hcmdpbi1ib3R0b206IDEwcHggIWltcG9ydGFudDtcbiAgICB9XG4gICAgLmltZy1uZXdfc2Nre1xuICAgICAgYmFja2dyb3VuZDogYmx1ZSB1cmwoXCIvYXNzZXRzL2ltYWdlcy9zY2tfZWR1LmpwZ1wiKSBjZW50ZXIvY292ZXI7XG4gICAgfVxuICAgIC5pbWctc2NrX2VkdXtcbiAgICAgIGJhY2tncm91bmQ6IGJsdWUgdXJsKFwiL2Fzc2V0cy9pbWFnZXMvbmV3X3Njay5qcGdcIikgY2VudGVyL2NvdmVyO1xuICAgIH1cbiAgfVxufVxuIiwiLyoqXG4gKiAgTkFWQkFSXG4gKi9cblxuLm5hdmJhcl9jb250YWluZXIge1xuICBhe1xuICAgICY6aG92ZXIsICY6YWN0aXZle1xuICAgICAgY29sb3I6ICRmb250X2NvbG9yX3doaXRlXG4gICAgfVxuICB9XG5cbiAgLm1kLWJ1dHRvbi5uYXZiYXJfaGlnaGxpZ2h0X2J1dHRvbiB7XG4gICAgYm9yZGVyOiAycHggc29saWQgJHllbGxvdztcbiAgICBjb2xvcjogJHllbGxvdztcbiAgICBib3JkZXItcmFkaXVzOiA1MHB4O1xuICAgIC8vcGFkZGluZzogOXB4IDExcHg7XG4gICAgLy93aWR0aDogMTA4cHg7XG4gICAgcGFkZGluZy10b3A6IC4wMXJlbTtcbiAgICBwYWRkaW5nLWJvdHRvbTogLjAxcmVtO1xuICAgIHRyYW5zaXRpb246IGNvbG9yIDAuNHMgY3ViaWMtYmV6aWVyKDAuMjUsIDAuOCwgMC4yNSwgMSksIGJhY2tncm91bmQtY29sb3IgMC40cyBjdWJpYy1iZXppZXIoMC4yNSwgMC44LCAwLjI1LCAxKTtcbiAgICAmOmhvdmVyLCAmOmFjdGl2ZSB7XG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yOiR5ZWxsb3cgIWltcG9ydGFudDtcbiAgICAgIGNvbG9yOiAkb2ZmX2JsYWNrICFpbXBvcnRhbnQ7XG4gICAgfVxuICB9XG5cbiAgLmxvZ29fbGluayB7XG4gICAgdGV4dC1hbGlnbjogY2VudGVyO1xuXG4gICAgLmxvZ29faWNvbiB7XG4gICAgICBoZWlnaHQ6IDMycHg7XG4gICAgICB3aWR0aDogMzJweDtcbiAgICB9XG4gIH1cblxuICAubmF2X2ljb24ge1xuICAgIG1hcmdpbi1yaWdodDogNHB4O1xuICAgIGhlaWdodDogMTRweCAhaW1wb3J0YW50O1xuICAgIHdpZHRoOiAxNHB4ICFpbXBvcnRhbnQ7XG4gICAgc3ZnIHtcbiAgICAgIGZpbGw6ICR5ZWxsb3c7XG4gICAgfVxuICB9XG5cbiAgLm5hdmJhcl9zaWdudXBfYnV0dG9uIHtcbiAgICAubWQtYnV0dG9uIHtcbiAgICAgIHBhZGRpbmc6IDJweCAwcHg7XG4gICAgICBib3JkZXI6IDJweCBzb2xpZCAkeWVsbG93O1xuICAgICAgY29sb3I6ICR5ZWxsb3c7XG4gICAgICBib3JkZXItcmFkaXVzOiA1MHB4O1xuICAgICAgd2lkdGg6IDc5cHg7XG4gICAgfVxuICAgIC5tZC1idXR0b246aG92ZXIge1xuICAgICAgYmFja2dyb3VuZC1jb2xvcjogJHllbGxvdztcbiAgICAgIGNvbG9yOiAkb2ZmX2JsYWNrO1xuICAgIH1cbiAgfVxuXG59XG5cbi5uYXZiYXJfYXZhdGFyX2ljb24ge1xuICB3aWR0aDogMzhweDtcbiAgaGVpZ2h0OiAzOHB4O1xuICBib3JkZXItcmFkaXVzOiAxOXB4O1xufVxuXG4vLyBNYWtlIHN1cmUgdGhlIERyb3Bkb3duIHN0YXlzIGJlbG93IHRoZSBidXR0b24gcHVzaGVkXG4ubWQtb3Blbi1tZW51LWNvbnRhaW5lci5tZC1hY3RpdmV7XG4gIG1hcmdpbi10b3A6IDUwcHggIWltcG9ydGFudDtcbn1cblxuIiwiLy8gLyoqXG4vLyAgKiAgU0VBUkNIIENPTVBPTkVOVFxuLy8gICovXG5cblxuI3NlYXJjaCBpbnB1dCB7XG4gICAgYmFja2dyb3VuZDogdXJsKC4uL2Fzc2V0cy9pbWFnZXMvc2VhcmNoX2ljb25fYmxhY2suc3ZnKTtcbiAgICBiYWNrZ3JvdW5kLXJlcGVhdDogbm8tcmVwZWF0O1xuICAgIGJhY2tncm91bmQtcG9zaXRpb246IDEwcHggMTJweDtcbiAgICBiYWNrZ3JvdW5kLXNpemU6IDE4cHg7XG4gICAgcGFkZGluZzogMHB4IDM1cHg7XG59XG5cbi5zZWFyY2hfcmVzdWx0cyB7XG4gIGRpc3BsYXk6IGZsZXg7XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIC5yZXN1bHRfaWNvbiB7XG4gICAgZmxleDogMCAwIDE4cHg7XG4gICAgbWFyZ2luLXJpZ2h0OiAxMnB4O1xuICAgIGhlaWdodDogMThweDtcbiAgfVxuICAucmVzdWx0X25hbWUge1xuICAgIG1hcmdpbi1yaWdodDogMTBweDtcbiAgfVxuICAucmVzdWx0X2xvY2F0aW9uIHtcbiAgICBjb2xvcjogJGJsdWU7XG4gIH1cbn1cbiIsIi8qXG4gIENPTlRBSU5FUlMgVVNFRCBPTiBTSE9XIEtJVCBTVEFURVxuICovXG5cbi5vdmVyX21hcHtcbiAgbWluLWhlaWdodDogMTIwcHg7XG59XG5cbi5raXRfZml4ZWQge1xuICB6LWluZGV4OiAkbWVkaXVtLXByaW9yaXR5O1xuICBwb3NpdGlvbjogcmVsYXRpdmU7XG59XG5cbi5raXRfZGF0YSB7XG4gIHdpZHRoOjEwMCU7XG4gIG1hcmdpbi10b3A6IDM3NnB4O1xuXG4gIHNlY3Rpb24ub3ZlcmxheXtcbiAgICB3aWR0aDoxMDAlO1xuICAgIGhlaWdodDoxMDAlO1xuICAgIHBvc2l0aW9uOmFic29sdXRlO1xuICAgIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xuICAgIHotaW5kZXg6IDM7XG4gICAgY29sb3I6ICRvZmZfYmxhY2s7XG4gICAgdGV4dC1hbGlnbjpjZW50ZXI7XG4gICAgZm9udC1zaXplOjIwcHg7XG4gICAgZm9udC13ZWlnaHQ6IGxpZ2h0ZXI7XG5cbiAgICBoMXtcbiAgICAgIGNvbG9yOiAkb2ZmX2JsYWNrO1xuICAgICAgZm9udC1zaXplOiAxLjhlbTtcbiAgICAgIG1hcmdpbjowO1xuICAgIH1cbiAgICBwe1xuICAgICAgY29sb3I6ICR0ZXJjaWFyeV9jb2xvcl9saWdodDtcbiAgICAgIG1hcmdpbjogMC41ZW07XG4gICAgfVxuICB9XG5cbiAgLmhpbnR7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogd2hpdGU7XG4gICAgdGV4dC1hbGlnbjpjZW50ZXI7XG4gICAgY29sb3I6JGZvbnRfY29sb3JfYmx1ZV9ncmV5O1xuXG4gICAgbWFyZ2luLXRvcDowcHg7XG5cbiAgICB6LWluZGV4OiAxMDtcbiAgICBwb3NpdGlvbjphYnNvbHV0ZTtcbiAgICBoZWlnaHQ6MTAwJTtcbiAgICB3aWR0aDoxMDAlO1xuICAgID4gcCB7XG4gICAgICBjb2xvcjogJGdyZXlfZGFya2VyO1xuICAgIH1cbiAgICAubWQtYnV0dG9ue1xuICAgICAgbWFyZ2luLXRvcDoyMHB4O1xuICAgIH1cbiAgfVxufVxuXG4ua2l0X2RhdGEsIC5raXRUYWdzX19zZWN0aW9ue1xuICAuc2hhZG93e1xuICAgIHRvcDogMHB4O1xuICAgIGhlaWdodDogMXB4O1xuICAgIGJhY2tncm91bmQtY29sb3I6IHJnYmEoMCwwLDAsMC4xKTtcbiAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gICAgYm94LXNoYWRvdzogMHB4IC0xcHggNnB4IDAgcmdiYSgwLDAsMCwwLjY1KTtcbiAgICB6LWluZGV4OiAyO1xuICB9XG59XG4ua2l0X3RhZ3Mtc2VsZWN0LWhlYWRlciB7XG4gIGhlaWdodDogNDhweDtcbiAgZGlzcGxheTogZmxleDtcbn1cbi5raXRfdGFncy1oZWFkZXItc2VhcmNoYm94IHtcbiAgYm9yZGVyOiBub25lO1xuICBib3JkZXItYm90dG9tOiAxcHggc29saWQgI2NjYztcbiAgcGFkZGluZy1sZWZ0OiAxMnB4O1xuICBoZWlnaHQ6IDEwMCU7XG4gIHdpZHRoOiAxMDAlO1xufVxuIiwiLypcbj09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5LSVQgQ09NTUVOVFMgTU9EVUxFXG49PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuKi9cbi5raXRfY29tbWVudHMge1xuICBwYWRkaW5nLWJvdHRvbTogMTAwcHg7XG59XG5cbi8qXG4gIENIQVJUIENPTVBPTkVOVFxuKi9cblxuLmNoYXJ0X2NvbnRhaW5lciB7XG4gIC8qbWluLXdpZHRoOiAzNDBweDsqL1xuICBtaW4taGVpZ2h0OiAzMTBweDtcbiAgLyp3aWR0aDogMTAwJTsqL1xuICAvKmhlaWdodDogOTAlOyovXG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcblxuICAuY2hhcnRfbGluZSB7XG4gICAgZmlsbDogbm9uZTtcbiAgICBzdHJva2Utd2lkdGg6IDJweDtcbiAgfVxuICAuY2hhcnRfYXJlYSB7XG4gICAgb3BhY2l0eTogMC4xO1xuICB9XG5cbiAgLmF4aXMgcGF0aCwuYXhpcyBsaW5lIHtcbiAgICBmaWxsOiBub25lO1xuICAgIHN0cm9rZTogJGdyZXlfZGFya2VyO1xuICAgIHN0cm9rZS13aWR0aDogMTtcbiAgICBzaGFwZS1yZW5kZXJpbmc6IGNyaXNwRWRnZXM7XG4gICAgZGlzcGxheTogbm9uZTtcbiAgfVxuICAuYXhpcyB7XG4gICAgZm9udC1zaXplOiAxMHB4O1xuICB9XG4gIC5heGlzLnlfbGVmdCB7XG4gICAgZmlsbDogJGdyZXk7XG4gIH1cbiAgLmF4aXMueV9yaWdodCB7XG4gICAgZmlsbDogJGdyZXk7XG4gIH1cbiAgLmF4aXMueCB7XG4gICAgZmlsbDogJGdyZXk7XG4gIH1cbiAgLmdyaWQge1xuICAgIC50aWNrIHtcbiAgICAgIHN0cm9rZTogJGdyZXk7XG4gICAgICBzdHJva2Utb3BhY2l0eTogMC42O1xuICAgICAgc2hhcGUtcmVuZGVyaW5nOiBjcmlzcEVkZ2VzO1xuICAgIH1cbiAgICBwYXRoIHtcbiAgICAgIHN0cm9rZS13aWR0aDogMDtcbiAgICB9XG4gIH1cbiAgLm92ZXJsYXkge1xuICAgIGZpbGw6IG5vbmU7XG4gICAgcG9pbnRlci1ldmVudHM6IGFsbDtcbiAgfVxuXG4gIC5mb2N1cyBjaXJjbGUge1xuICAgIGZpbGw6IHJnYigzLCAzNywgNDUpO1xuICAgIHN0cm9rZS13aWR0aDogMnB4O1xuICB9XG4gIC50ZXh0X2hvdmVyX2NvbnRhaW5lciB7XG4gICAgYm9yZGVyOiAxcHggc29saWQgJGdyZXlfZGFya2VyO1xuICB9XG4gIC5wb3B1cF92YWx1ZSB7XG4gICAgZmlsbDogJGJhY2tncm91bmRfYmx1ZV9kYXJrX2NoYXJ0O1xuICAgIGNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2RhcmtfY2hhcnQ7XG4gICAgZm9udC1zaXplOiAxOHB4O1xuICAgIGZvbnQtd2VpZ2h0OiBib2xkO1xuICB9XG4gIC5wb3B1cF9kYXRlIHtcbiAgICBmaWxsOiAkZ3JleV9kYXJrZXI7XG4gICAgZm9udC1zaXplOiAxMHB4O1xuICAgIGZvbnQtd2VpZ2h0OiBib2xkO1xuICB9XG4gIC5wb3B1cF91bml0IHtcbiAgICBmaWxsOiAkZ3JleTtcbiAgICBtYXJnaW4tbGVmdDogMTBweDtcbiAgfVxufVxuXG4uc3RpY2tOYXYge1xuICBwb3NpdGlvbjogZml4ZWQ7XG4gIHRvcDogMHB4O1xufVxuXG4ub3ZlcmxheS1raXRpbmZve1xuICBiYWNrZ3JvdW5kOiB3aGl0ZTtcbiAgaGVpZ2h0OiAxMDAlO1xuICB3aWR0aDogMTAwJTtcbiAgcG9zaXRpb246YWJzb2x1dGU7XG4gIHotaW5kZXg6OTk5O1xufVxuXG5cbi8qXG4gIEtJVCBNRU5VIE1PRFVMRVxuICovXG5zZWN0aW9uIC5raXRfbWVudSB7XG4gIHdpZHRoOiAxMDAlO1xuICB6LWluZGV4OiAkbW9yZV9wcmlvcml0eTtcbiAgYmFja2dyb3VuZC1jb2xvcjogJGdyZXlfbGlnaHRlcjtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xuXG4gIC5jb250YWluZXIge1xuICAgIHdpZHRoOiAxMDAlO1xuICAgIGhlaWdodDogMzNweDtcbiAgfVxuXG4gIC5raXRfdGltZSB7XG4gICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgIC8vbGVmdDogLTQwcHg7XG4gICAgdGV4dC1hbGlnbjogY2VudGVyO1xuICAgIGNvbG9yOiAkZ3JleV9kYXJrZXI7XG4gICAgbGV0dGVyLXNwYWNpbmc6IDAuNXB4O1xuICAgIGZvbnQtc2l6ZTogMTRweDtcbiAgICB0ZXh0LXRyYW5zZm9ybTogdXBwZXJjYXNlO1xuICB9XG4gIC5raXRfdXNlciB7XG4gICAgbWFyZ2luLWxlZnQ6MjBweDtcbiAgICAvL2Zsb2F0OiBsZWZ0O1xuICAgIGxpbmUtaGVpZ2h0OiAzMHB4O1xuICAgIC8vbWFyZ2luOiAwIDMycHggMCAyMHB4O1xuICAgIGNvbG9yOiAkZ3JleV9kYXJrZXI7XG4gICAgZm9udC13ZWlnaHQ6IGJvbGQ7XG4gICAgZm9udC1zaXplOiAxM3B4O1xuXG4gICAgaW1nIHtcbiAgICAgIGJvcmRlci1yYWRpdXM6NTAlO1xuICAgICAgZmxvYXQ6bGVmdDtcbiAgICAgIHdpZHRoOiAxOHB4O1xuICAgICAgaGVpZ2h0OiAxOHB4O1xuICAgICAgbWFyZ2luLXJpZ2h0OiA4cHg7XG4gICAgICBtYXJnaW4tdG9wOiA1cHg7XG4gICAgfVxuXG4gICAgYSB7XG4gICAgICBjb2xvcjogJGdyZXlfZGFya2VyO1xuICAgICAgdGV4dC1kZWNvcmF0aW9uOiBub25lO1xuICAgIH1cbiAgfVxuICAua2l0X25hbWUge1xuICAgIG1hcmdpbi1sZWZ0OjMwcHg7XG4gICAgLy9mbG9hdDogbGVmdDtcbiAgICBsaW5lLWhlaWdodDogMzBweDtcbiAgICBmb250LXNpemU6IDEzcHg7XG4gICAgZm9udC13ZWlnaHQ6IGJvbGQ7XG4gICAgY29sb3I6ICRncmV5X2RhcmtlcjtcbiAgICAvL21hcmdpbjogMCAzMnB4IDAgMDtcblxuICAgIG1kLWljb24ge1xuICAgICAgZmxvYXQ6bGVmdDtcbiAgICAgIHdpZHRoOiAxNHB4O1xuICAgICAgaGVpZ2h0OiAxNHB4O1xuICAgICAgbWFyZ2luLXJpZ2h0OiA4cHg7XG4gICAgICBtYXJnaW4tdG9wOiA0cHg7XG4gICAgICBzdmcgeyBmaWxsOiAkeWVsbG93OyB9XG4gICAgfVxuICAgIHNwYW4ge1xuICAgICAgdmVydGljYWwtYWxpZ246IG1pZGRsZTtcbiAgICB9XG4gIH1cblxuICAua2l0X2JhdHRlcnkge1xuICAgIG1hcmdpbi1sZWZ0OjMwcHg7XG4gICAgLy9mbG9hdDogbGVmdDtcbiAgICBsaW5lLWhlaWdodDogMzBweDtcbiAgICBjb2xvcjogJGdyZXlfZGFya2VyO1xuICAgIGZvbnQtc2l6ZTogMTJweDtcbiAgICBmb250LXdlaWdodDogYm9sZDtcblxuICAgIG1kLWljb24ge1xuICAgICAgbWFyZ2luLWJvdHRvbTogMnB4O1xuICAgICAgd2lkdGg6IDE2cHg7XG4gICAgICBtYXJnaW4tcmlnaHQ6IDhweDtcbiAgICAgIHN2ZyB7XG4gICAgICAgIGZpbGw6ICRncmV5X2RhcmtlcjtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICAua2l0LXNob3ctcmF3e1xuICAgIG1hcmdpbi1yaWdodDogNnB4O1xuICB9XG5cbiAgLmtpdC1yYXctdG9nZ2xle1xuICAgIHZlcnRpY2FsLWFsaWduOiBtaWRkbGU7XG4gIH1cblxuICAvKlNob3cgcmF3IFRvZ2dsZSovXG4gIC5zd2l0Y2gge1xuICAgIHZlcnRpY2FsLWFsaWduOiBtaWRkbGU7XG4gICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgIGRpc3BsYXk6IGlubGluZS1ibG9jaztcbiAgICB3aWR0aDogMzBweDtcbiAgICBoZWlnaHQ6IDE1cHg7XG4gIH1cblxuICAuc3dpdGNoIGlucHV0IHsgXG4gICAgb3BhY2l0eTogMDtcbiAgICB3aWR0aDogMDtcbiAgICBoZWlnaHQ6IDA7XG4gIH1cblxuICAuc2xpZGVyIHtcbiAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgdmVydGljYWwtYWxpZ246IG1pZGRsZTtcbiAgICBjdXJzb3I6IHBvaW50ZXI7XG4gICAgdG9wOiAwO1xuICAgIGxlZnQ6IDA7XG4gICAgcmlnaHQ6IDA7XG4gICAgYm90dG9tOiAwO1xuICAgIGJhY2tncm91bmQtY29sb3I6ICNjY2M7XG4gICAgLXdlYmtpdC10cmFuc2l0aW9uOiAuNHM7XG4gICAgdHJhbnNpdGlvbjogLjRzO1xuICB9XG5cbiAgLnNsaWRlcjpiZWZvcmUge1xuICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICBjb250ZW50OiBcIlwiO1xuICAgIGhlaWdodDogMTFweDtcbiAgICB3aWR0aDogMTFweDtcbiAgICBsZWZ0OiAzcHg7XG4gICAgYm90dG9tOiAycHg7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogd2hpdGU7XG4gICAgLXdlYmtpdC10cmFuc2l0aW9uOiAuNHM7XG4gICAgdHJhbnNpdGlvbjogLjRzO1xuICB9XG5cbiAgaW5wdXQ6Y2hlY2tlZCArIC5zbGlkZXIge1xuICAgIGJhY2tncm91bmQtY29sb3I6ICR5ZWxsb3c7XG4gIH1cblxuICBpbnB1dDpmb2N1cyArIC5zbGlkZXIge1xuICAgIGJveC1zaGFkb3c6IDAgMCAxcHggIzIxOTZGMztcbiAgfVxuXG4gIGlucHV0OmNoZWNrZWQgKyAuc2xpZGVyOmJlZm9yZSB7XG4gICAgLXdlYmtpdC10cmFuc2Zvcm06IHRyYW5zbGF0ZVgoMTRweCk7XG4gICAgLW1zLXRyYW5zZm9ybTogdHJhbnNsYXRlWCgxNHB4KTtcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVgoMTRweCk7XG4gIH1cblxuICAvKiBSb3VuZGVkIHNsaWRlciAqL1xuICAuc2xpZGVyLnJvdW5kIHtcbiAgICBib3JkZXItcmFkaXVzOiAzNHB4O1xuICB9XG5cbiAgLnNsaWRlci5yb3VuZDpiZWZvcmUge1xuICAgIGJvcmRlci1yYWRpdXM6IDUwJTtcbiAgfVxuXG4gIC5raXQtc2hvdy1yYXctdGV4dHtcbiAgICB2ZXJ0aWNhbC1hbGlnbjogbWlkZGxlO1xuICAgIGNvbG9yOiAkZ3JleV9kYXJrZXI7XG4gICAgZm9udC1zaXplOiAxMnB4O1xuICAgIGZvbnQtd2VpZ2h0OiBib2xkO1xuICB9XG5cbiAgLmtpdF9uYXZiYXIge1xuICAgIG1hcmdpbi1yaWdodDogMjBweDtcblxuICAgIC8vIEljb25zIGFyZSBpbnNpZGUgbWQtYnV0dG9uXG4gICAgbWQtaWNvbiB7XG4gICAgICB3aWR0aDogMTRweDtcbiAgICAgIHN2ZyB7XG4gICAgICAgIGZpbGw6ICRncmV5X2RhcmtlcjtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAubWQtYnV0dG9uIHtcbiAgICAgIG1hcmdpbjowO1xuICAgIH1cblxuICAgIGEubWQtYnV0dG9uOmhvdmVyLCAuYnV0dG9uX2FjdGl2ZSAge1xuICAgICAgYm9yZGVyLWJvdHRvbTogMnB4IHNvbGlkICRvZmZfYmxhY2s7XG4gICAgfVxuXG4gIH1cblxufVxuXG4ua2l0X292ZXJ2aWV3IHtcbiAgd2lkdGg6IDEwMCU7XG4gIGhlaWdodDogOTZweDtcbiAgei1pbmRleDogJG1lZGl1bV9wcmlvcml0eTtcbiAgYmFja2dyb3VuZC1jb2xvcjogJGdyZXlfbGlnaHRlc3Q7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcblxuICAuYnV0dG9uX3Njcm9sbCB7XG4gICAgbWFyZ2luOiAwcHg7XG4gICAgLy9wb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgLy90b3A6IDA7XG4gICAgLy93aWR0aDogNjBweDtcbiAgICAvL2hlaWdodDogOTZweDtcbiAgICBiYWNrZ3JvdW5kOiAjY2NjO1xuICB9XG5cbiAgLmJ1dHRvbl9zY3JvbGxfbGVmdCB7XG4gICAgbGVmdDogMDtcbiAgfVxuXG4gIC5idXR0b25fc2Nyb2xsX3JpZ2h0IHtcbiAgICByaWdodDogMDtcbiAgfVxuXG4gIC5zZW5zb3JzX2NvbnRhaW5lciB7XG4gICAgd2lkdGg6IDkwJTtcbiAgICBoZWlnaHQ6IDk2cHg7XG4gICAgbWFyZ2luOiAwIGF1dG87XG4gICAgb3ZlcmZsb3cteDogaGlkZGVuO1xuICAgIG92ZXJmbG93LXk6IGhpZGRlbjtcbiAgfVxuXG4gIEBtZWRpYSAocG9pbnRlcjogY29hcnNlKSB7XG4gICAgLnNlbnNvcnNfY29udGFpbmVyIHtcbiAgICAgIG92ZXJmbG93LXg6IGF1dG87XG4gICAgfVxuICB9XG5cbiAgLnNlbnNvcl9jb250YWluZXIge1xuICAgIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgICBjb2xvcjogJGdyZXlfZGFya2VyO1xuICAgIGRpc3BsYXk6IGlubGluZS10YWJsZTtcbiAgICBjdXJzb3I6IHBvaW50ZXI7XG4gICAgdGV4dC1hbGlnbjogY2VudGVyO1xuICAgIG91dGxpbmU6bm9uZTtcblxuICAgIHB7XG4gICAgICBtYXJnaW46IDBweDtcbiAgICAgIHBhZGRpbmctYm90dG9tOiAxNXB4O1xuICAgICAgZm9udC1zaXplOiAwLjhlbTtcbiAgICAgIGZvbnQtd2VpZ2h0OiBub3JtYWw7XG4gICAgfVxuXG4gICAgLmNsZWFyIHtcbiAgICAgIGNsZWFyOiBib3RoO1xuICAgIH1cblxuICAgIC5zZW5zb3JfcmlnaHQge1xuICAgICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgICAgcmlnaHQ6IDE1cHg7XG4gICAgICBib3R0b206IDM4cHg7XG4gICAgICBtYXgtd2lkdGg6IDMwcHg7XG4gICAgICAuc2Vuc29yX3VuaXQge1xuICAgICAgICBtYXJnaW4tdG9wOjhweDtcbiAgICAgICAgZm9udC1zaXplOiAxNHB4O1xuICAgICAgfVxuICAgICAgLnNlbnNvcl9hcnJvdyB7XG4gICAgICAgIG1hcmdpbi10b3A6LTE1cHg7XG4gICAgICAgIHdpZHRoOiAxNHB4O1xuICAgICAgICBoZWlnaHQ6IDdweDtcbiAgICAgICAgJi5hcnJvd191cCBzdmcgeyBmaWxsOiAkZ3JlZW47IH1cbiAgICAgICAgJi5hcnJvd19kb3duIHN2ZyB7IGZpbGw6ICRyZWQ7IH1cbiAgICAgICAgJi5lcXVhbCBzdmcgeyBmaWxsOiAkYmx1ZV9saWdodDsgfVxuICAgICAgfVxuICAgIH1cbiAgICAuc2Vuc29yX3ZhbHVlIHtcbiAgICAgIG1hcmdpbi10b3A6IDEwcHg7XG4gICAgICBwYWRkaW5nLWxlZnQ6IDUwcHg7XG4gICAgICBwYWRkaW5nLXJpZ2h0OiA1MHB4O1xuICAgIH1cbiAgICAuc2Vuc29yX3ZhbHVlX251bGwge1xuICAgICAgb3BhY2l0eTogMC4xNTtcbiAgICB9XG4gICAgLnNlbnNvcl9pY29uIHtcbiAgICAgIHdpZHRoOiAyMnB4O1xuICAgICAgaGVpZ2h0OiAyMnB4O1xuICAgICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgICAgYm90dG9tOiA0NXB4O1xuICAgICAgbGVmdDogMjVweDtcbiAgICAgIHN2ZyB7XG4gICAgICAgIGZpbGw6ICRncmV5X2RhcmtlcjtcbiAgICAgIH1cbiAgICB9XG4gICAgJjpob3ZlciB7XG4gICAgICBvcGFjaXR5OiAwLjg7XG4gICAgfVxuICAgICYuc2VsZWN0ZWQge1xuICAgICAgY29sb3I6ICRvZmZfYmxhY2s7XG4gICAgICAuc2Vuc29yX2ljb24gc3ZnIHtcbiAgICAgICAgZmlsbDogJG9mZl9ibGFjaztcbiAgICAgIH1cbiAgICB9XG4gIH1cbn1cblxuLyoqXG4gKiAgS0lUIERFVEFJTFNcbiAqXG4gKi9cblxuIC5raXRfZGV0YWlsc19ub3RBdXRoIHtcbiAgIC8vIEJ1ZzogdGhpcyB3aWxsIGNyZWF0ZSBhbiBleHRyYSBnYXAgb24gbW9iaWxlIGRldmljZXNcbiAgLy93aWR0aDogNTAwcHg7XG4gfVxuLmtpdF9kZXRhaWxzX25vdEF1dGhfdGl0bGUge1xuICBtYXJnaW46IDA7XG4gfVxuLmtpdF9kZXRhaWxzX25vdEF1dGhfc3ViaGVhZGVyIHtcbiAgY29sb3I6ICRncmV5X2Rhcmtlcjtcbn1cbi5raXRfZGV0YWlsc19ub3RBdXRoX2J1dHRvbiB7XG4gIGZsb2F0OiByaWdodDtcbn1cbi5raXRfZGV0YWlsZWQge1xuICAvKmhlaWdodDogMTAwMHB4OyovXG4gIHotaW5kZXg6ICRtZWRpdW1fcHJpb3JpdHk7XG4gIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xuICBwb3NpdGlvbjogcmVsYXRpdmU7XG5cbiAgcC5kZXNjcmlwdGlvbiB7XG4gICAgbGluZS1oZWlnaHQ6IGluaGVyaXQ7XG4gICAgZm9udC13ZWlnaHQ6IGxpZ2h0ZXI7XG4gICAgd2lkdGg6IDc1JTtcbiAgfVxuXG4gIC5raXRfdGltZU9wdHN7XG4gICAgbWFyZ2luOiAwO1xuICAgIGZvbnQtc2l6ZTogMS4zZW07XG4gIH1cblxuICAua2l0X2NoYXJ0IHtcbiAgICBtaW4taGVpZ2h0OiAzNDBweDtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiB3aGl0ZTtcbiAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG5cbiAgICAuaGludHtcbiAgICAgIGJhY2tncm91bmQtY29sb3I6ICRvZmZfYmxhY2s7XG4gICAgICBvcGFjaXR5OiAwLjY7XG4gICAgfVxuXG4gICAgLmNvbnRhaW5lciB7XG4gICAgICBwYWRkaW5nOiAzN3B4O1xuICAgICAgaGVpZ2h0OiAxMDAlO1xuXG4gICAgICAua2l0X2NoYXJ0X2xlZnQge1xuICAgICAgICBjb2xvcjogJG9mZl9ibGFjaztcblxuICAgICAgICAuc2Vuc29yX3NlbGVjdCB7XG4gICAgICAgICAgLnNlbnNvcl9pY29uX3NlbGVjdGVkIHtcbiAgICAgICAgICAgIGZsb2F0OmxlZnQ7XG4gICAgICAgICAgICBtYXJnaW4tcmlnaHQ6IDZweDtcbiAgICAgICAgICAgIHBhZGRpbmctdG9wOjVweDtcbiAgICAgICAgICAgIHN2ZyB7XG4gICAgICAgICAgICAgIGZpbGw6ICRvZmZfYmxhY2s7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICAgIG1kLXNlbGVjdCB7XG4gICAgICAgICAgICBtYXJnaW46IDA7XG4gICAgICAgICAgICAubWQtc2VsZWN0LXZhbHVle1xuICAgICAgICAgICAgICAubWQtdGV4dHtcbiAgICAgICAgICAgICAgICAuc2VsZWN0X2ltYWdle1xuICAgICAgICAgICAgICAgICAgZGlzcGxheTpub25lO1xuXG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIC5zZW5zb3JfZGVzY3JpcHRpb24ge1xuICAgICAgICAgIC8qLy9jb2xvcjogJGJhY2tncm91bmRfYmx1ZV9saWdodDsqL1xuICAgICAgICAgIC8qLy9oZWlnaHQ6IDgwcHg7Ki9cbiAgICAgICAgICAvKi8vbWFyZ2luLXRvcDogMHB4OyAvLzMwOyovXG4gICAgICAgICAgLyovL3Bvc2l0aW9uOiByZWxhdGl2ZTsqL1xuXG4gICAgICAgICAgaDYge1xuICAgICAgICAgICAgbWFyZ2luOiAwO1xuICAgICAgICAgICAgbWFyZ2luLWJvdHRvbTozcHg7XG4gICAgICAgICAgICBmb250LXNpemU6IDFlbTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBzbWFsbCB7XG4gICAgICAgICAgICBsaW5lLWhlaWdodDogMWVtO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIGEge1xuICAgICAgICAgICAgbWFyZ2luLWxlZnQ6IDEwcHg7XG4gICAgICAgICAgICB0ZXh0LWRlY29yYXRpb246IG5vbmU7XG4gICAgICAgICAgICBjb2xvcjogJGdyZXlfZGFya2VyO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIC5zZW5zb3JfZGVzY3JpcHRpb25fZnVsbCB7XG4gICAgICAgICAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgICAgICAgICB0b3A6IC01MHB4O1xuICAgICAgICAgICAgbWFyZ2luLXRvcDogMTBweDtcbiAgICAgICAgICAgIGRpc3BsYXk6IG5vbmU7XG4gICAgICAgICAgICBvdmVyZmxvdzogdmlzaWJsZTtcbiAgICAgICAgICAgIHotaW5kZXg6IDIwMDtcbiAgICAgICAgICAgIC8vIGJvcmRlcjogMXB4IHNvbGlkIGdyZXk7XG4gICAgICAgICAgICBiYWNrZ3JvdW5kLWNvbG9yOiByZ2JhKDI1NSwgMjU1LCAyNTUsIDAuOCk7XG4gICAgICAgICAgICBjb2xvcjogJGJhY2tncm91bmRfYmx1ZV9kYXJrX2NoYXJ0O1xuICAgICAgICAgICAgYm9yZGVyLXJhZGl1czogMnB4O1xuICAgICAgICAgICAgbGluZS1oZWlnaHQ6IDEuNWVtO1xuICAgICAgICAgICAgcGFkZGluZzogMTBweDtcbiAgICAgICAgICAgIGJveC1zaGFkb3c6IDAgMXB4IDVweCByZ2JhKDAsIDAsIDAsIDAuNjUpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIC5zZW5zb3JfZGF0YSB7XG4gICAgICAgICAgLnNlbnNvcl92YWx1ZSB7XG4gICAgICAgICAgICBmb250LXNpemU6IDJlbTtcbiAgICAgICAgICAgIGZvbnQtd2VpZ2h0OiBib2xkO1xuICAgICAgICAgIH1cbiAgICAgICAgICAuc2Vuc29yX3VuaXQge1xuICAgICAgICAgICAgZm9udC1zaXplOiAuOGVtO1xuICAgICAgICAgICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgICAgICAgICAgdG9wOiAtOHB4O1xuICAgICAgICAgIH1cbiAgICAgICAgICAuc2Vuc29yX2RhdGFfaWNvbiB7XG4gICAgICAgICAgICB3aWR0aDogMTRweDtcbiAgICAgICAgICAgIGhlaWdodDogMTRweDtcbiAgICAgICAgICAgIG1hcmdpbi1sZWZ0OiA4cHg7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIC5zZW5zb3JfZGF0YV9kZXNjcmlwdGlvbiB7XG4gICAgICAgICAgZm9udC1zaXplOiAxMnB4O1xuICAgICAgICAgIGRpc3BsYXk6IG5vbmU7XG4gICAgICAgICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgICAgICAgIC13ZWJraXQtdHJhbnNmb3JtOiB0cmFuc2xhdGUoNXB4LCA1cHgpO1xuICAgICAgICAgIC1tb3otdHJhbnNmb3JtOiB0cmFuc2xhdGUoNXB4LCA1cHgpO1xuICAgICAgICAgIC1tcy10cmFuc2Zvcm06IHRyYW5zbGF0ZSg1cHgsIDVweCk7XG4gICAgICAgICAgLW8tdHJhbnNmb3JtOiB0cmFuc2xhdGUoNXB4LCA1cHgpO1xuICAgICAgICAgIHRyYW5zZm9ybTogdHJhbnNsYXRlKDVweCwgNXB4KTtcbiAgICAgICAgICBiYWNrZ3JvdW5kLWNvbG9yOiByZ2JhKDI1NSwgMjU1LCAyNTUsIDAuOCk7XG4gICAgICAgICAgY29sb3I6ICRiYWNrZ3JvdW5kX2JsdWVfZGFya19jaGFydDtcbiAgICAgICAgICBib3JkZXItcmFkaXVzOiAycHg7XG4gICAgICAgICAgbGluZS1oZWlnaHQ6IDEuNWVtO1xuICAgICAgICAgIHBhZGRpbmc6IDRweCA2cHg7XG4gICAgICAgICAgYm94LXNoYWRvdzogMCAxcHggNXB4IHJnYmEoMCwgMCwgMCwgMC42NSk7XG4gICAgICAgICAgbWF4LXdpZHRoOiAxNzVweDtcbiAgICAgICAgfVxuICAgICAgICAuc2Vuc29yX2NvbXBhcmUge1xuICAgICAgICAgIGZvbnQtc2l6ZTogLjhlbTtcbiAgICAgICAgICBwYWRkaW5nLXRvcDozMHB4O1xuICAgICAgICAgIG1hcmdpbi10b3A6IDEwcHg7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgLmtpdF9jaGFydF9yaWdodCB7XG4gICAgICAgIC8qLy9kaXNwbGF5OiBpbmxpbmUtYmxvY2s7Ki9cbiAgICAgICAgLyovL2hlaWdodDogMTAwJTsqL1xuXG4gICAgICAgIC5jaGFydF9uYXZpZ2F0aW9uIHtcbiAgICAgICAgICBoZWlnaHQ6IDEwJTtcbiAgICAgICAgICBmb250LXNpemU6IDExcHg7XG4gICAgICAgICAgPiBkaXYge1xuICAgICAgICAgICAgbWFyZ2luLWxlZnQ6IDIwcHg7XG4gICAgICAgICAgfVxuICAgICAgICAgIGlucHV0IHtcbiAgICAgICAgICAgIG1hcmdpbi1sZWZ0OiAxMHB4O1xuICAgICAgICAgICAgcGFkZGluZy1sZWZ0OiAxMHB4O1xuICAgICAgICAgICAgZm9udC1zaXplOiAxMnB4O1xuICAgICAgICAgIH1cbiAgICAgICAgICAua2l0X3RpbWUge1xuICAgICAgICAgICAgbWQtaWNvbiB7XG4gICAgICAgICAgICAgIHdpZHRoOiAxMXB4O1xuICAgICAgICAgICAgICBoZWlnaHQ6IDExcHg7XG4gICAgICAgICAgICAgIG1hcmdpbi1yaWdodDogOHB4O1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC5waWNrZXJfY29udGFpbmVyIHtcbiAgICAgICAgICAgIG1hcmdpbi1sZWZ0OiAyMHB4O1xuICAgICAgICAgIH1cblxuICAgICAgICAgIEBtZWRpYSBvbmx5IHNjcmVlbiBhbmQgKG1heC1kZXZpY2Utd2lkdGg6IDEwNTBweCkge1xuICAgICAgICAgICAgLmNoYXJ0X25hdmlnYXRpb257XG4gICAgICAgICAgICAgIGp1c3RpZnktY29udGVudDogc3BhY2UtYXJvdW5kICFpbXBvcnRhbnQ7XG4gICAgICAgICAgICAgIG1hcmdpbjogNXB4ICFpbXBvcnRhbnQ7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICAucGlja2VyX2NvbnRhaW5lciB7XG4gICAgICAgICAgICAgIG1hcmdpbjogMCA1cHggMCA1cHggIWltcG9ydGFudDtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG5cbiAgICAgICAgICBAbWVkaWEgb25seSBzY3JlZW4gYW5kIChtYXgtZGV2aWNlLXdpZHRoOiA5NjBweCkge1xuXG4gICAgICAgICAgICAucGlja2VyX2NvbnRhaW5lciB7XG4gICAgICAgICAgICAgIGRpc3BsYXk6IGZsZXggIWltcG9ydGFudDtcbiAgICAgICAgICAgICAgYWxpZ24tY29udGVudDogbWlkZGxlICFpbXBvcnRhbnQ7XG4gICAgICAgICAgICAgIG1hcmdpbjogMTBweCAwIDAgMCAhaW1wb3J0YW50O1xuICAgICAgICAgICAgICB3aWR0aDogMTAwJSAhaW1wb3J0YW50O1xuICAgICAgICAgICAgICBqdXN0aWZ5LWNvbnRlbnQ6IHNwYWNlLWJldHdlZW4gIWltcG9ydGFudDtcbiAgICAgICAgICAgICAgYWxpZ24taXRlbXM6IGNlbnRlciAhaW1wb3J0YW50O1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAud29yZF9waWNrZXJ7XG4gICAgICAgICAgICAgIGRpc3BsYXk6IGlubGluZS1ibG9jayAhaW1wb3J0YW50O1xuICAgICAgICAgICAgICB3aWR0aDogMTAwJSAhaW1wb3J0YW50O1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgLmNoYXJ0X21vdmUge1xuICAgICAgICAgICAgICBkaXNwbGF5OiBmbGV4ICFpbXBvcnRhbnQ7XG4gICAgICAgICAgICAgIHdpZHRoOiAxMDAlICFpbXBvcnRhbnQ7XG4gICAgICAgICAgICAgIGp1c3RpZnktY29udGVudDogc3BhY2UtYmV0d2VlbiAhaW1wb3J0YW50O1xuICAgICAgICAgICAgICBtYXJnaW4tdG9wOiAxNXB4ICFpbXBvcnRhbnQ7XG5cbiAgICAgICAgICAgICAgLmNoYXJ0X21vdmVfYnV0dG9uIHtcbiAgICAgICAgICAgICAgICB3aWR0aDogMTAwJSAhaW1wb3J0YW50O1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgLmNoYXJ0X21vdmUge1xuICAgICAgICAgICAgLmNoYXJ0X21vdmVfYnV0dG9uIHtcbiAgICAgICAgICAgICAgbWFyZ2luOiAwcHggLTJweCAwIC0ycHg7XG4gICAgICAgICAgICAgIHdpZHRoOiA0MHB4O1xuICAgICAgICAgICAgICBoZWlnaHQ6IDI4cHg7XG4gICAgICAgICAgICAgIGRpc3BsYXk6IGlubGluZS1mbGV4O1xuICAgICAgICAgICAgICBib3JkZXI6IDFweCBzb2xpZCAkZ3JleV9kYXJrZXI7XG4gICAgICAgICAgICAgIGJhY2tncm91bmQtY29sb3I6ICRncmV5X2xpZ2h0ZXN0O1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgLmNoYXJ0X21vdmVfcmlnaHQge1xuICAgICAgICAgICAgICAvLyBtYXJnaW4tbGVmdDogNHB4O1xuICAgICAgICAgICAgICBib3JkZXItdG9wLWxlZnQtcmFkaXVzOiAwcHg7XG4gICAgICAgICAgICAgIGJvcmRlci1ib3R0b20tbGVmdC1yYWRpdXM6IDBweDtcbiAgICAgICAgICAgICAgc3ZnIHtcbiAgICAgICAgICAgICAgICBmaWxsOiAkZ3JleV9kYXJrZXI7XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgJjpob3ZlciB7XG4gICAgICAgICAgICAgICAgYmFja2dyb3VuZC1jb2xvcjogJGdyZXlfZGFya2VyO1xuICAgICAgICAgICAgICAgIHN2ZyB7XG4gICAgICAgICAgICAgICAgICBmaWxsOiAkZ3JleV9saWdodGVzdDtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIC5jaGFydF9tb3ZlX2xlZnQge1xuICAgICAgICAgICAgICBzdmcge1xuICAgICAgICAgICAgICAgIGZpbGw6ICRncmV5X2RhcmtlcjtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAmOmhvdmVyIHtcbiAgICAgICAgICAgICAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkZ3JleV9kYXJrZXI7XG4gICAgICAgICAgICAgICAgc3ZnIHtcbiAgICAgICAgICAgICAgICAgIGZpbGw6ICRncmV5X2xpZ2h0ZXN0O1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgICAgbWQtaWNvbiB7XG4gICAgICAgICAgICAgIHdpZHRoOiA1LjVweDtcbiAgICAgICAgICAgICAgaGVpZ2h0OiA5cHg7XG4gICAgICAgICAgICAgIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgICAgICAgICAgICAgdG9wOiAwcHg7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuICB9XG4gIC5raXRfZGV0YWlscyB7XG4gICAgY29sb3I6ICRiYWNrZ3JvdW5kX2JsdWVfZGFya19uYXZiYXI7XG4gIH1cbiAgLmtpdF9kZXRhaWxzX2NvbnRlbnQge1xuICAgIHN2ZyB7XG4gICAgICBmaWxsOiAkb2ZmX2JsYWNrO1xuICAgIH1cblxuICAgIC5raXRfZGV0YWlsc19sYWJlbHN7XG4gICAgICBtYXJnaW4tdG9wOjMwcHg7XG4gICAgfVxuICB9XG4gIC5raXRfZGV0YWlsc19uYW1lIHtcbiAgICBtYXJnaW46IDVweCAwIDIwcHggMDtcbiAgfVxuICAua2l0X2RldGFpbHNfdHlwZSB7XG4gICAgPiBzcGFuIHtcbiAgICAgIG1hcmdpbi1yaWdodDogMTVweDtcbiAgICB9XG4gIH1cbiAgLmtpdF9kZXRhaWxzX3R5cGVfbG9jYXRpb24gc3ZnIHtcbiAgICBmaWxsOiAkc2Vjb25kYXJ5X2NvbG9yO1xuICB9XG4gIC5raXRfZGV0YWlsc192ZXJzaW9uIHtcbiAgICBzcGFuIHtcbiAgICAgIGNvbG9yOiAkc2Vjb25kYXJ5X2NvbG9yX2xpZ2h0O1xuICAgICAgZm9udC13ZWlnaHQ6IGJvbGQ7XG4gICAgfVxuICB9XG4gIC5raXRfZGV0YWlsc19pZCB7XG4gICAgc3BhbiB7XG4gICAgICBjb2xvcjogJHNlY29uZGFyeV9jb2xvcl9saWdodDtcbiAgICAgIGZvbnQtd2VpZ2h0OiBib2xkO1xuICAgIH1cbiAgfVxuXG4gIC5raXRfZGV0YWlsc19kZXNjcmlwdGlvbiB7XG4gICAgcGFkZGluZy10b3A6IDRweDtcbiAgfVxuICAua2l0X2RldGFpbHNfZGVzY3JpcHRpb25fdGl0bGUge1xuICAgIGZvbnQtc2l6ZTogMTZweDtcbiAgICBtYXJnaW4tYm90dG9tOiAxNXB4O1xuICB9XG4gIC5raXRfZGV0YWlsc19kZXNjcmlwdGlvbl9jb250ZW50IHtcbiAgICBmb250LXNpemU6IDEycHg7XG4gIH1cbiAgLmtpdF9kZXRhaWxzX21hbmFnZV9idXR0b25zIHtcbiAgICBtYXJnaW4tYm90dG9tOiAzMnB4O1xuICB9XG4gIC5raXRfZGV0YWlsc19idXR0b25fZGVsZXRlIHtcbiAgICBjb2xvcjogJHJlZDtcbiAgfVxuICAua2l0X2RldGFpbHNfbWFuYWdlX2RhdGEge1xuICAgIC5kYXRhX2NvbnRhaW5lciB7XG4gICAgfVxuICB9XG4gIC5raXRfZGV0YWlsc19rZXkge1xuICAgIGZsb2F0OiBsZWZ0O1xuICAgIGhlaWdodDogMjZweDtcbiAgfVxuICAubWQtYnV0dG9uLmtpdF9kZXRhaWxzX2tleV9yZWZyZXNoQnV0dG9uOmhvdmVyLCAubWQtYnV0dG9uLmtpdF9kZXRhaWxzX2tleV9yZWZyZXNoQnV0dG9uOmZvY3VzIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjREJEQkRCO1xuICB9XG4gIC5raXRfZGV0YWlsZWRfaWNvbl9jb250ZW50IHtcbiAgICB3aWR0aDogMTRweDtcbiAgICBoZWlnaHQ6IDE0cHg7XG4gICAgbWFyZ2luLXJpZ2h0OiA0cHg7XG4gICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgIHRvcDogLTFweDtcbiAgICBzdmcge1xuICAgICAgZmlsbDogJGdyZXlfZGFya2VyO1xuICAgIH1cbiAgfVxuICAua2l0X2RldGFpbGVkX2ljb25fdGl0bGUge1xuICAgIHdpZHRoOiAxOHB4O1xuICAgIGhlaWdodDogMThweDtcbiAgICBtYXJnaW4tcmlnaHQ6IDhweDtcbiAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gICAgdG9wOiAtMnB4O1xuICB9XG4gIC5raXRfZGV0YWlsZWRfdGl0bGVfY29udGFpbmVyIHtcbiAgICBwYWRkaW5nOiAyMHB4IDA7XG4gIH1cbiAgLmtpdF9kZXRhaWxlZF90aXRsZSB7XG4gICAgZm9udC1zaXplOiAxNnB4O1xuICAgIGRpc3BsYXk6IGlubGluZTtcbiAgfVxuICAua2l0X2RldGFpbGVkX2NvbnRlbnRfY29udGFpbmVyIHtcbiAgICBtYXJnaW46IDYycHggMzdweCAwcHg7XG5cbiAgICAua2l0X2RldGFpbHNfY29udGVudF9tYWluIHtcbiAgIC8vIEJ1ZzogdGhpcyB3aWxsIGNyZWF0ZSBhbiBleHRyYSBnYXAgb24gbW9iaWxlIGRldmljZXNcbiAgICAvLyAgd2lkdGg6IDUwMHB4O1xuICAgIH1cbiAgfVxuICAua2l0X293bmVyIHtcbiAgICBjb2xvcjogJGJhY2tncm91bmRfYmx1ZV9kYXJrX25hdmJhcjtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjRjVGNUY1O1xuICAgIHBhZGRpbmctYm90dG9tOjQwcHg7XG5cbiAgICAuY29udGFpbmVyIHtcbiAgICAgIHdpZHRoOiA5NSU7XG4gICAgICBtYXJnaW46IDAgYXV0bztcbiAgICB9XG4gICAgLmtpdF9vd25lcl91c2VybmFtZUxpbmsge1xuICAgICAgdGV4dC1kZWNvcmF0aW9uOiBub25lO1xuICAgIH1cbiAgICAua2l0X293bmVyX3VzZXJuYW1lVGV4dCB7XG4gICAgICBjb2xvcjogJG9mZl9ibGFjaztcbiAgICAgIG1hcmdpbjogNTBweCAwIDIwcHggMDtcbiAgICB9XG4gICAgLmtpdF9vd25lcl9sb2NhdGlvbiB7XG4gICAgICBjb2xvcjogJHNlY29uZGFyeV9jb2xvcjtcbiAgICAgIGZvbnQtd2VpZ2h0OiBib2xkO1xuICAgIH1cbiAgICAua2l0X293bmVyX3VybCBhIHtcbiAgICAgIHRleHQtZGVjb3JhdGlvbjogbm9uZTtcbiAgICAgIGNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2RhcmtfbmF2YmFyO1xuICAgIH1cbiAgICAua2l0X293bmVyX2tpdHNfdGl0bGUge1xuICAgICAgbWFyZ2luLWJvdHRvbTogNTBweDtcbiAgICB9XG4gICAgLmtpdF9vd25lcl9raXRzX2xpc3Qge1xuICAgICAgYm9yZGVyOiAxcHggc29saWQgI0VBRUNGMTtcbiAgICAgIGhlaWdodDogODJweDtcbiAgICAgIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xuICAgIH1cbiAgICAua2l0X293bmVyX2tpdHNfbGlzdF9hdmF0YXIge1xuICAgICAgZmxvYXQ6IGxlZnQ7XG4gICAgICB3aWR0aDogNDRweDtcbiAgICAgIGhlaWdodDogNDRweDtcbiAgICAgIG1hcmdpbi1yaWdodDogMyU7XG4gICAgICBtYXJnaW4tdG9wOiA5cHg7XG4gICAgfVxuICAgIC5raXRfb3duZXJfa2l0c19saXN0X2NvbnRlbnQge1xuICAgICAgZmxvYXQ6IGxlZnQ7XG4gICAgICBtYXJnaW4tdG9wOiA1cHg7XG4gICAgfVxuICAgIC5raXRfb3duZXJfa2l0c19saXN0X2RhdGEge1xuICAgICAgZm9udC1zaXplOiAxNnB4O1xuICAgICAgc3BhbiB7XG4gICAgICAgIG1hcmdpbi1yaWdodDogMTVweDtcbiAgICAgIH1cbiAgICB9XG4gICAgLmtpdF9vd25lcl9raXRzX2xpc3Rfc3RhdGUge1xuICAgICAgZGlzcGxheTogaW5saW5lO1xuICAgICAgZm9udC1zaXplOiAxMnB4O1xuICAgICAgYm9yZGVyLXJhZGl1czogMjBweDtcbiAgICAgIHBhZGRpbmc6IDZweCAxMXB4O1xuICAgICAgbWFyZ2luOiAxNXB4O1xuICAgIH1cbiAgICAua2l0X293bmVyX2tpdHNfbGlzdF9uZXZlcl9wdWJsaXNoZWQge1xuICAgICAgYmFja2dyb3VuZC1jb2xvcjogJHllbGxvdztcbiAgICB9XG4gICAgLmtpdF9vd25lcl9raXRzX2xpc3Rfbm90X2NvbmZpZ3VyZWQge1xuICAgICAgYmFja2dyb3VuZC1jb2xvcjogJHJlZDtcbiAgICB9XG4gICAgLmtpdF9vd25lcl9raXRzX2xpc3RfY29uZmlnIHtcbiAgICAgIGZsb2F0OiByaWdodDtcbiAgICAgIG1hcmdpbi10b3A6IDEzcHg7XG4gICAgfVxuICAgIC5raXRfb3duZXJfa2l0c19saXN0X2NvbmZpZ19pY29uIHtcbiAgICAgIHdpZHRoOiAyMnB4O1xuICAgICAgaGVpZ2h0OiAyMnB4O1xuICAgIH1cbiAgfVxufVxuXG4uc2Vuc29yX3ZhbHVlIHtcbiAgZm9udC1zaXplOiAzNHB4O1xuICBmb250LWZhbWlseTogS2FuaXQ7XG4gIGZvbnQtd2VpZ2h0OiA3MDA7XG59XG5cbi8qKi9cblxuLyogWyFdIExlYXZlIGFuaW1hdGlvbiAobmctbGVhdmUgKSBhcmUgZGlzYWJsZWRcbmJlY2F1c2UgdGhleSBjb25mbGljdCB3aXRoIGVudGVyIGFuaW1hdGlvbnMgKi9cblxuXG4uc2Vuc29yX2FuaW1hdGlvbi5uZy1lbnRlciB7XG4gIHRyYW5zaXRpb246MXMgZWFzZS1vdXQgYWxsO1xufVxuLnNlbnNvcl9hbmltYXRpb24ubmctZW50ZXIge1xuICB0cmFuc2Zvcm06IHNjYWxlKDEuMDUpO1xuICAvLyBvcGFjaXR5OiAwO1xufVxuLnNlbnNvcl9hbmltYXRpb24ubmctZW50ZXItYWN0aXZlIHtcbiAgdHJhbnNmb3JtOiBzY2FsZSgxKTtcbiAgLy8gb3BhY2l0eTogMTtcbn1cblxuLyogLnNlbnNvcl9hbmltYXRpb24ubmctbGVhdmV7XG4gIHRyYW5zaXRpb246MXMgZWFzZSBhbGw7XG59XG4uc2Vuc29yX2FuaW1hdGlvbi5uZy1sZWF2ZXtcbiAgb3BhY2l0eTogMDtcbn1cbi5zZW5zb3JfYW5pbWF0aW9uLm5nLWxlYXZlLWFjdGl2ZXtcbiAgb3BhY2l0eTogMTtcbn0gKi9cblxuLyogVGltZSBhbmltYXRpb24gY3VycmVudGx5IG5vdCBpbiB1c2UgKi9cblxuLnRpbWVfYW5pbWF0aW9uLm5nLWVudGVyIHtcbiAgdHJhbnNpdGlvbjoxcyBlYXNlLW91dCBhbGw7XG59XG4udGltZV9hbmltYXRpb24ubmctZW50ZXIge1xuICB0cmFuc2Zvcm06IHNjYWxlKDEuMDUpO1xufVxuLnRpbWVfYW5pbWF0aW9uLm5nLWVudGVyLWFjdGl2ZSB7XG4gIHRyYW5zZm9ybTogc2NhbGUoMSk7XG59XG5cbi8qIC50aW1lX2FuaW1hdGlvbi5uZy1sZWF2ZSB7XG4gIHRyYW5zaXRpb246MXMgZWFzZSBhbGw7XG59XG4udGltZV9hbmltYXRpb24ubmctbGVhdmUge1xuICBvcGFjaXR5OiAwO1xufVxuLnRpbWVfYW5pbWF0aW9uLm5nLWxlYXZlLWFjdGl2ZSB7XG4gIG9wYWNpdHk6IDE7XG59ICovXG5cbi5iYXRfYW5pbWF0aW9uLm5nLWVudGVyIHtcbiAgdHJhbnNpdGlvbjoxcyBlYXNlLW91dCBhbGw7XG59XG4uYmF0X2FuaW1hdGlvbi5uZy1lbnRlciB7XG4gIHRyYW5zZm9ybTogc2NhbGUoMS4wNSk7XG59XG4uYmF0X2FuaW1hdGlvbi5uZy1lbnRlci1hY3RpdmUge1xuICB0cmFuc2Zvcm06IHNjYWxlKDEpO1xufVxuXG4vKiAuYmF0X2FuaW1hdGlvbi5uZy1sZWF2ZSB7XG4gIHRyYW5zaXRpb246MnMgZWFzZSBhbGw7XG59XG4uYmF0X2FuaW1hdGlvbi5uZy1sZWF2ZSB7XG4gIG9wYWNpdHk6IDA7XG59XG4uYmF0X2FuaW1hdGlvbi5uZy1sZWF2ZS1hY3RpdmUge1xuICBvcGFjaXR5OiAxO1xufSAqL1xuIiwiLypcbiAgTU9EQUwgU1RPUkVcbiovXG5cbi5zdG9yZV9pdGVtRGVzY3JpcHRpb24ge1xuICBtYXJnaW4tbGVmdDoyMHB4O1xuICB1bCB7XG4gICAgbGlzdC1zdHlsZS1pbWFnZTogdXJsKCcuLi9hc3NldHMvaW1hZ2VzL2NoZWNrX2NpcmNsZS5zdmcnKTtcbiAgICBsaXN0LXN0eWxlLXBvc2l0aW9uOiBpbmhlcml0O1xuICAgIGZvbnQtc2l6ZTogLjhlbTtcbiAgICBsaW5lLWhlaWdodDogMS44ZW07XG4gIH1cbn1cbiIsIi5jc3ZfZmlsZV9pdGVtIHtcbiAgZGlzcGxheTogZmxleDtcbiAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgbWQtY2hlY2tib3gge1xuICAgIG1hcmdpbi1ib3R0b206IDBweDtcbiAgfVxuICBzcGFuIHtcbiAgICBtYXJnaW4tcmlnaHQ6IDIwcHhcbiAgfVxufVxuXG4ubGlzdC1zaGFkb3cge1xuICBib3gtc2hhZG93OiAwcHggMXB4IDNweCAwcHggcmdiYSgwLDAsMCwwLjIpO1xuICBtZC1saXN0LWl0ZW0ge1xuICAgIGJvcmRlci1ib3R0b206IHNvbGlkIDFweCAkZ3JleV9saWdodGVyO1xuICB9XG4gIG1kLWxpc3QtaXRlbTpsYXN0LWNoaWxkIHtcbiAgICBib3JkZXItYm90dG9tOiBub25lO1xuICB9XG59XG5cbi8vIC5tZC1iYXIuYmctZ3JlZW4ge1xuLy8gICBiYWNrZ3JvdW5kLWNvbG9yOiAkZ3JlZW47XG4vLyB9XG5cbi5sYWJlbC1ncmV5IHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJGdyZXk7XG59XG5cbi50aXRsZS10aW1lbGluZSB7XG4gIGZvbnQtc2l6ZTogMzBweDtcbn1cblxuLyogVGhpcyBpcyB0ZW1wb3JhcnkgKi9cblxuLnVwbG9hZC1jc3Yge1xuICAgIC5jb250YWluZXIge1xuICAgICAgLy8gcGFkZGluZy1ib3R0b206MjBweDtcbiAgICAgIG1hcmdpbi1yaWdodDogYXV0bztcbiAgICAgIG1hcmdpbi1sZWZ0OiBhdXRvO1xuICAgIH1cblxuICAgIC5jc3ZfY29udGVudHtcbiAgICAgIG1hcmdpbi10b3A6IDI0MHB4O1xuICAgICAgbWluLWhlaWdodDogNTB2aDtcbiAgICB9XG5cbiAgICBAbWVkaWEgKG1pbi13aWR0aDogNzY4cHgpIHtcbiAgICAgICAgLmNvbnRhaW5lciB7XG4gICAgICAgICAgICB3aWR0aDogNzUwcHg7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBAbWVkaWEgKG1pbi13aWR0aDogOTkycHgpIHtcbiAgICAgICAgLmNvbnRhaW5lciB7XG4gICAgICAgICAgICB3aWR0aDogOTcwcHg7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBAbWVkaWEgKG1pbi13aWR0aDogMTIwMHB4KSB7XG4gICAgICAgIC5jb250YWluZXIge1xuICAgICAgICAgICAgd2lkdGg6IDExNzBweDtcbiAgICAgICAgfVxuICAgIH1cbn1cbiIsIi8qKlxuICogIEJBQ0tEUk9QIENPTVBPTkVOVFxuICovXG5cbi8vIFBvcHVwIGJhY2tkcm9wXG5cbm1kLWNvbnRlbnQubWQtb3BhcXVlIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogcmdiYSgzMCwgMzAsIDMwLCAwLjYpO1xufVxuXG4vL0xvYWRpbmcgYmFja2Ryb3BcblxubWQtY29udGVudC5tZC1tYWluQmFja2Ryb3Age1xuICBkaXNwbGF5OiBmbGV4O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBqdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjtcbiAgd2lkdGg6IDEwMHZ3O1xuICBoZWlnaHQ6IDEwMHZoO1xuICAuYmFja2Ryb3BfaWNvbiB7XG4gICAgd2lkdGg6IDEwMXB4O1xuICAgIGhlaWdodDogMTAxcHg7XG4gIH1cbn1cblxuLy8gc3RhdGUgY2hhbmdlIGJhY2tkcm9wXG5tZC1jb250ZW50Lm1kLXN0YXRlQ2hhbmdlQmFja2Ryb3Age1xuICB6LWluZGV4OiA5OTk7XG4gIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgYmFja2dyb3VuZC1jb2xvcjogdHJhbnNwYXJlbnQ7XG4gIHdpZHRoOiAxMDB2dztcbiAgaGVpZ2h0OiAxMDB2aDtcbn1cblxuLmJhY2tkcm9wX2ljb24ge1xuICBzdmcge1xuICAgIG92ZXJmbG93OiB2aXNpYmxlO1xuICAgIGZpbGw6ICRncmV5X2RhcmtlcjtcbiAgfVxuICAuc2MtY2lyY2xlLWFycm93IHtcbiAgICAtd2Via2l0LXRyYW5zZm9ybS1vcmlnaW46IDUwJSA1My44JTtcbiAgICAtbW96LXRyYW5zZm9ybS1vcmlnaW46IDUwJSA1My44JTtcbiAgICAtby10cmFuc2Zvcm0tb3JpZ2luOiA1MCUgNTMuOCU7XG4gICAgdHJhbnNmb3JtLW9yaWdpbjogNTAlIDUzLjglO1xuICAgIC13ZWJraXQtYW5pbWF0aW9uOnNwaW4gM3MgbGluZWFyIGluZmluaXRlO1xuICAgIC1tb3otYW5pbWF0aW9uOnNwaW4gM3MgbGluZWFyIGluZmluaXRlO1xuICAgIGFuaW1hdGlvbjpzcGluIDNzIGxpbmVhciBpbmZpbml0ZTtcbiAgfVxufVxuXG5ALW1vei1rZXlmcmFtZXMgc3BpbiB7IDEwMCUgeyAtbW96LXRyYW5zZm9ybTogcm90YXRlKDM2MGRlZyk7IH0gfVxuQC13ZWJraXQta2V5ZnJhbWVzIHNwaW4geyAxMDAlIHsgLXdlYmtpdC10cmFuc2Zvcm06IHJvdGF0ZSgzNjBkZWcpOyB9IH1cbkBrZXlmcmFtZXMgc3BpbiB7IDEwMCUgeyAtd2Via2l0LXRyYW5zZm9ybTogcm90YXRlKDM2MGRlZyk7IHRyYW5zZm9ybTpyb3RhdGUoMzYwZGVnKTsgfSB9XG4iLCJzZWN0aW9uIC5zdGlja01lbnUge1xuICBwb3NpdGlvbjogYWJzb2x1dGUgIWltcG9ydGFudDtcbi8vICB0b3A6IDY0cHg7XG59XG5cbi5zdGlja05hdiB7XG4gIHBvc2l0aW9uOiBmaXhlZDtcbiAgdG9wOiAwcHg7XG4gIGJvcmRlci1ib3R0b206MXB4IHNvbGlkICM2NjY7XG59XG4iXX0= */
diff --git a/styles/app-79f1132de1.css b/styles/app-79f1132de1.css
new file mode 100644
index 00000000..fef3f841
--- /dev/null
+++ b/styles/app-79f1132de1.css
@@ -0,0 +1,3257 @@
+@import url(https://fonts.googleapis.com/css?family=Roboto+Condensed:400,300,300italic,400italic,700,700italic|Roboto:400,700,700italic,400italic);
+@import url("https://fonts.googleapis.com/css?family=Kanit:400,500,600,700,900");
+.border-black {
+ border: 3px solid #1C1C1C; }
+
+.border-white {
+ border: 3px solid white; }
+
+.color-white {
+ color: white !important; }
+
+.color-dropdown {
+ color: #7E7E7E !important; }
+
+.color-black {
+ color: #1C1C1C; }
+
+.color-blue {
+ color: #0019FF; }
+
+.color-cyan {
+ color: #0019FF; }
+
+.color-green {
+ color: #00E597; }
+ .color-green svg {
+ fill: #00E597; }
+
+.color-red {
+ color: #FF3D4C; }
+ .color-red svg {
+ fill: #FF3D4C !important; }
+
+.float-left {
+ float: left; }
+
+.float-right {
+ float: right; }
+
+.d-flex {
+ display: flex; }
+
+.full-width {
+ width: 100%; }
+
+.max-width-500px {
+ max-width: 500px; }
+
+.min-height-80 {
+ min-height: 80%; }
+
+.font-roboto-condensed {
+ font-family: 'Roboto Condensed'; }
+
+.font-kanit {
+ font-family: 'Kanit'; }
+
+.text-center {
+ text-align: center; }
+
+.text-left {
+ text-align: left; }
+
+.bg-white {
+ background-color: #fff; }
+
+.bg-black {
+ background-color: black; }
+
+.bg-green {
+ background-color: #00E597; }
+
+.bg-blue {
+ background-color: #0019FF; }
+
+.bg-red {
+ background-color: #FF3D4C !important; }
+
+.bg-red-light {
+ background-color: rgba(255, 61, 76, 0.2); }
+
+.bg-yellow {
+ background-color: #FFC100; }
+
+.bg-grey {
+ background-color: #D2D3D5 !important; }
+
+.bg-grey-lightest {
+ background-color: #F9F9FB !important; }
+
+.outline {
+ outline: 1px solid red; }
+
+.btn-outline-blue {
+ border: 2px solid #0019FF;
+ color: #0019FF;
+ padding: 10px 29px; }
+ .btn-outline-blue svg {
+ fill: #0019FF; }
+
+.btn-outline-blue:hover {
+ background: #0019FF !important;
+ color: white; }
+ .btn-outline-blue:hover svg {
+ fill: white; }
+
+.btn-outline-white {
+ border: 2px solid white;
+ color: white;
+ padding: 10px 29px; }
+
+.btn-outline-white:hover {
+ background: white !important;
+ color: #1C1C1C; }
+
+.btn-outline-white-blue {
+ border: 2px solid white;
+ color: white;
+ padding: 10px 29px; }
+
+.btn-outline-white-blue:hover {
+ background: #0019FF !important; }
+
+.btn-outline-yellow {
+ border: 2px solid #FFC100;
+ color: #FFC100;
+ padding: 10px 29px; }
+
+.btn-outline-yellow:hover {
+ background: #FFC100 !important;
+ color: #1C1C1C; }
+
+.btn-yellow {
+ border: 2px solid #FFC100;
+ background: #FFC100;
+ color: #1C1C1C;
+ padding: 12px 29px; }
+
+.btn-yellow:hover {
+ background: #1C1C1C !important;
+ color: #FFC100; }
+
+.btn-black-outline {
+ border: 2px solid #1C1C1C;
+ color: #1C1C1C;
+ padding: 10px 29px; }
+
+.btn-black-outline:hover {
+ background-color: #1C1C1C !important;
+ color: white; }
+
+.btn-blue {
+ border: 2px solid #0019FF;
+ background: #0019FF;
+ color: white;
+ padding: 12px 29px; }
+
+.btn-blue:hover {
+ color: #0019FF;
+ background: #ECECEE !important; }
+
+.btn-white-blue {
+ border: 2px solid white;
+ background: none;
+ color: white;
+ padding: 12px 29px; }
+
+.btn-white-blue:hover {
+ color: #0019FF;
+ background: white !important;
+ border: 2px solid #0019FF; }
+
+.btn-round-new {
+ border-radius: 40px;
+ font-family: 'Roboto Condensed';
+ font-size: 16px;
+ line-height: 40px;
+ font-weight: 600;
+ letter-spacing: 0.0325em;
+ padding: 0px 35px;
+ transition: box-shadow 0.4s cubic-bezier(0.25, 0.8, 0.25, 1), background-color 0.4s cubic-bezier(0.25, 0.8, 0.25, 1); }
+
+.btn-round {
+ border: 0;
+ border-radius: 25px;
+ padding: 5px 35px;
+ font-size: 18px; }
+
+.btn-cyan {
+ background-color: #0019FF;
+ color: white;
+ border: 2px solid #0019FF; }
+
+.btn-cyan:hover {
+ color: #0019FF;
+ background: white !important;
+ border: 2px solid #0019FF; }
+
+.btn-full {
+ display: block;
+ width: 100%;
+ border-radius: 0px;
+ margin: 0px;
+ padding: 12px 29px; }
+
+.btn-small {
+ min-width: unset !important;
+ min-height: unset !important; }
+
+.m-0 {
+ margin: 0; }
+
+.m-10 {
+ margin: 10px; }
+
+.ml-0 {
+ margin-left: 0px; }
+
+.ml-15 {
+ margin-left: 15px; }
+
+.mt-10 {
+ margin-top: 10px; }
+
+.mt-20 {
+ margin-top: 20px; }
+
+.mt-30 {
+ margin-top: 30px; }
+
+.mt-50 {
+ margin-top: 50px; }
+
+.mr-10 {
+ margin-right: 10px; }
+
+.mr-20 {
+ margin-right: 20px; }
+
+.mr-30 {
+ margin-right: 30px; }
+
+.mr-50 {
+ margin-right: 50px; }
+
+.mb-10 {
+ margin-bottom: 10px; }
+
+.mb-20 {
+ margin-bottom: 20px; }
+
+.mb-30 {
+ margin-bottom: 30px; }
+
+.mb-50 {
+ margin-bottom: 50px; }
+
+.ml-10 {
+ margin-left: 10px; }
+
+.ml-20 {
+ margin-left: 20px; }
+
+.ml-30 {
+ margin-left: 30px; }
+
+.ml-50 {
+ margin-left: 50px; }
+
+.my-20 {
+ margin-top: 20px !important;
+ margin-bottom: 20px !important; }
+
+.mw-100 {
+ max-width: 100%; }
+
+.p-0 {
+ padding: 0; }
+
+.p-10 {
+ padding: 10px; }
+
+.p-20 {
+ padding: 20px; }
+
+.p-30 {
+ padding: 30px; }
+
+.p-40 {
+ padding: 40px; }
+
+.p-50 {
+ padding: 50px; }
+
+.p-60 {
+ padding: 60px; }
+
+.pl-20 {
+ padding-left: 20px; }
+
+.pt-80 {
+ padding-top: 80px; }
+
+.py-40 {
+ padding-top: 40px;
+ padding-bottom: 40px; }
+
+.px-20 {
+ padding-left: 20px;
+ padding-right: 20px; }
+
+.px-40 {
+ padding-left: 40px;
+ padding-right: 40px; }
+
+.img-circle {
+ border-radius: 50%; }
+
+footer p {
+ font-size: 12px !important;
+ line-height: 12px !important; }
+
+.absolute {
+ position: absolute; }
+
+.relative {
+ position: relative; }
+
+.s-48 {
+ width: 48px;
+ height: 48px; }
+
+html {
+ scroll-behavior: smooth; }
+
+body.active {
+ overflow-y: hidden;
+ overflow-x: hidden;
+ position: fixed; }
+
+.grey-waves {
+ background: grey url("/assets/images/sck_bg.png");
+ background-repeat: repeat;
+ background-size: 65px; }
+
+.bar {
+ display: block;
+ width: 25px;
+ height: 3px;
+ margin: 5px auto;
+ transition: all 0.5s ease-in-out;
+ background-color: white; }
+
+.hamburger {
+ display: none; }
+
+.logo-box {
+ background-color: #1C1C1C;
+ width: 100%;
+ display: flex;
+ justify-content: center;
+ align-content: center; }
+
+.logo-box h1 {
+ color: white !important;
+ font-weight: 800 !important;
+ font-size: 50px;
+ padding: 15px;
+ text-align: left; }
+
+.card-text {
+ padding: 30px 50px;
+ align-self: center; }
+
+.navigation {
+ display: flex;
+ flex-direction: row;
+ justify-content: space-between;
+ width: auto;
+ align-items: center;
+ padding: 6px 0% 6px 10%; }
+
+.display-none {
+ display: none !important; }
+
+.display-block {
+ display: block !important; }
+
+.new-landing-page {
+ display: flex;
+ overflow: hidden;
+ flex-direction: column;
+ padding-bottom: 60px; }
+ .new-landing-page h1, .new-landing-page h2, .new-landing-page h3, .new-landing-page h4, .new-landing-page h5, .new-landing-page h6 {
+ letter-spacing: 0.0325em;
+ font-family: 'Kanit';
+ font-weight: 900;
+ color: #1C1C1C; }
+ .new-landing-page h1 {
+ line-height: 55px;
+ margin: 0px 0px; }
+ .new-landing-page h2 {
+ font-size: 43px;
+ line-height: 40px;
+ margin: 0px 0px; }
+ .new-landing-page h3 {
+ font-size: 30px;
+ line-height: 30px;
+ margin-top: 10px;
+ margin-bottom: 10px; }
+ .new-landing-page p {
+ color: #1C1C1C;
+ font-family: 'Roboto Condensed';
+ font-size: 20px;
+ line-height: 24px; }
+ .new-landing-page .p-header {
+ font-size: 24px; }
+ .new-landing-page a {
+ text-decoration: none !important; }
+ .new-landing-page .landing-menu-btn {
+ border-radius: 40px;
+ font-family: 'Roboto Condensed';
+ font-size: 16px;
+ line-height: 40px;
+ font-weight: 600;
+ letter-spacing: 0.0325em;
+ padding: 0px 30px;
+ color: white;
+ transition: box-shadow 0.4s cubic-bezier(0.25, 0.8, 0.25, 1), background-color 0.4s cubic-bezier(0.25, 0.8, 0.25, 1); }
+ .new-landing-page .btn-justify-left {
+ margin-left: 0px; }
+ .new-landing-page .landing-menu .btn-round-new.md-button.btn-kit {
+ margin-left: 38px; }
+ .new-landing-page .landing-menu-btn:hover {
+ background: none;
+ color: #FFC100; }
+ .new-landing-page .sc-logo {
+ height: 37px;
+ top: 5px; }
+ .new-landing-page .sc-off-cta-platform {
+ position: fixed;
+ top: 0;
+ z-index: 3;
+ display: flex;
+ flex-direction: row;
+ flex-wrap: nowrap;
+ width: calc(100% - 20px);
+ justify-content: space-between;
+ align-items: center;
+ background-color: #1C1C1C;
+ margin: 10px 10px; }
+ .new-landing-page #get-your-kit {
+ scroll-margin-top: 78px; }
+ .new-landing-page .lighter {
+ max-width: 10%;
+ transition: all 0.3s ease-in-out;
+ flex-direction: column;
+ padding: 5px;
+ opacity: 90%; }
+ .new-landing-page .lighter:hover {
+ opacity: 100%; }
+ .new-landing-page .landing-menu {
+ display: flex;
+ flex-direction: row;
+ flex-wrap: wrap;
+ width: -webkit-fit-content;
+ width: -moz-fit-content;
+ width: fit-content;
+ align-self: center;
+ justify-content: space-between;
+ padding: 6px 10% 6px 0%; }
+ .new-landing-page .img-new_sck {
+ min-height: 30vw;
+ background: blue url("/assets/images/landing/Legacy-SCK.gif") center/cover; }
+ .new-landing-page .img-sck_edu {
+ min-height: 30vw;
+ background: blue url("/assets/images/landing/smart-citizen-data.jpg") center/cover; }
+ .new-landing-page .img-sck_com {
+ background: blue url("/assets/images/landing/local-communites.jpg") center/cover; }
+ .new-landing-page .img-research {
+ background: blue url("/assets/images/landing/research-institutions.jpg") center/cover; }
+ .new-landing-page .img-governm {
+ background: blue url("/assets/images/landing/educators.jpg") center/cover; }
+ .new-landing-page .img-platform {
+ background: blue url("/assets/images/landing/SCK-Platform-mobile.jpg") center/cover; }
+ .new-landing-page .img-docs {
+ background: blue url("/assets/images/landing/smartcitizen-docs.jpg") top/cover; }
+ .new-landing-page .img-kits-complete {
+ background: blue url("/assets/images/landing/smart-citizen-2-2-kit.gif") center/cover; }
+ .new-landing-page .img-kits-research {
+ background: blue url("/assets/images/landing/SCK-customized-projects.jpg") center/cover; }
+ .new-landing-page .tile {
+ padding: 50px 50px;
+ border-bottom: 3px solid #1C1C1C;
+ border-right: 3px solid #1C1C1C; }
+ .new-landing-page .tile-image.tile {
+ min-height: 300px;
+ padding: 0 !important; }
+ .new-landing-page .tile-left {
+ border-left: 3px solid #1C1C1C; }
+ .new-landing-page .tile-top {
+ border-top: 3px solid #1C1C1C; }
+ .new-landing-page .text-funding {
+ text-align: left; }
+ .new-landing-page .pd-60 {
+ padding-top: 60px; }
+ .new-landing-page .text-left {
+ text-align: left;
+ justify-content: left; }
+ .new-landing-page .video-section {
+ background: url("/assets/images/landing/smartcitizen-workshop.jpg") center/cover;
+ position: relative;
+ top: 0px;
+ height: 100vh;
+ overflow: hidden;
+ border: 3px solid #1C1C1C;
+ box-sizing: border-box; }
+ .new-landing-page .mobileonly {
+ display: none; }
+ .new-landing-page .header-section {
+ margin: 60px 10%;
+ max-width: 1200px; }
+ .new-landing-page .heading-over-video {
+ position: absolute;
+ z-index: 2;
+ bottom: 80px;
+ max-width: 600px;
+ width: 50%;
+ background-color: white;
+ border: 3px solid #1C1C1C; }
+ .new-landing-page .heading-over-video img {
+ padding-bottom: 40px; }
+ .new-landing-page .heading-over-video.right-card {
+ right: 10%; }
+ .new-landing-page #learn-more {
+ margin: 0px auto 60px auto;
+ max-width: 1200px;
+ padding: 0 10%; }
+ .new-landing-page .banner-section {
+ background: url("/assets/images/landing/SCK-Platform.jpg") center/cover;
+ position: relative;
+ height: 100vh;
+ width: 100%;
+ overflow: hidden;
+ border: 3px solid #1C1C1C;
+ box-sizing: border-box; }
+ .new-landing-page .card {
+ box-shadow: 10px 10px 0px -3px #FFC100, 10px 10px 0px 0px #1C1C1C; }
+ @media only screen and (max-width: 1200px) {
+ .new-landing-page #learn-more {
+ padding: 0; }
+ .new-landing-page .card {
+ box-shadow: 0px 0px; } }
+ @media only screen and (max-width: 1010px) {
+ .new-landing-page h1 {
+ font-size: 42px;
+ line-height: 42px; }
+ .new-landing-page h2 {
+ font-size: 28px;
+ line-height: 28px; }
+ .new-landing-page h3 {
+ font-size: 26px;
+ line-height: 26px; }
+ .new-landing-page p {
+ font-size: 18px;
+ line-height: 22px; }
+ .new-landing-page .sc-logo {
+ left: 5%; }
+ .new-landing-page .heading-over-video img {
+ padding-bottom: 00px; }
+ .new-landing-page #get-your-kit {
+ scroll-margin-top: 80px; } }
+ @media only screen and (max-width: 950px) {
+ .new-landing-page .mobileonly {
+ display: block;
+ border: 3px solid #1C1C1C;
+ border-bottom: none; }
+ .new-landing-page .mobileonly.img-platform {
+ border-top: none; }
+ .new-landing-page .sc-off-cta-platform {
+ justify-self: center;
+ align-self: start;
+ align-items: start;
+ text-align: left;
+ flex-wrap: wrap;
+ transition: all 0.3s ease-in-out; }
+ .new-landing-page .navigation.active {
+ padding: 16px 0% 6px 11%; }
+ .new-landing-page .external-links {
+ display: flex;
+ flex-direction: column; }
+ .new-landing-page .hamburger {
+ display: block;
+ cursor: pointer;
+ align-self: center;
+ padding: 6px 10% 6px 0%; }
+ .new-landing-page .hamburger.active {
+ display: block;
+ cursor: pointer;
+ align-self: center;
+ padding: 16px 11% 6px 0%; }
+ .new-landing-page .hamburger.active .bar:nth-child(2) {
+ opacity: 0; }
+ .new-landing-page .hamburger.active .bar:nth-child(1) {
+ -webkit-transform: translateY(8px) rotate(45deg);
+ transform: translateY(8px) rotate(45deg); }
+ .new-landing-page .hamburger.active .bar:nth-child(3) {
+ -webkit-transform: translateY(-8px) rotate(-45deg);
+ transform: translateY(-8px) rotate(-45deg); }
+ .new-landing-page .landing-menu {
+ display: none;
+ justify-content: space-between;
+ width: 100%;
+ height: 40%;
+ padding: 0px 10%; }
+ .new-landing-page .landing-menu.active {
+ display: flex;
+ align-self: start;
+ transition: all .3s ease-in-out;
+ flex-direction: column;
+ justify-content: space-between;
+ padding-bottom: 10%; }
+ .new-landing-page .landing-menu .md-button.landing-menu-btn {
+ text-align: left;
+ padding: 0px 0px; }
+ .new-landing-page .landing-menu .btn-round-new.md-button.btn-kit {
+ align-self: end;
+ justify-self: end; }
+ .new-landing-page .sc-off-cta-platform.active {
+ height: 100%;
+ width: 100%;
+ justify-content: space-between;
+ align-content: space-between;
+ transition: all .3s ease-in-out;
+ margin: 0%; }
+ .new-landing-page .sc-logo {
+ padding: 10px 0px; }
+ .new-landing-page #get-your-kit {
+ scroll-margin-top: 85px; }
+ .new-landing-page .layout-xs-column {
+ flex-direction: column; }
+ .new-landing-page .layout-xs-column > .flex-xs-100 {
+ flex: 1 1 100%;
+ max-width: 100%;
+ max-height: 100%;
+ box-sizing: border-box; }
+ .new-landing-page .border-xs-top {
+ border-top: 3px solid #1C1C1C;
+ border-left: 3px solid #1C1C1C;
+ border-right: 3px solid #1C1C1C;
+ border-bottom: 3px solid #1C1C1C; }
+ .new-landing-page .border-xs-bottom {
+ border-top: unset;
+ border-left: 3px solid #1C1C1C;
+ border-right: 3px solid #1C1C1C;
+ border-bottom: 3px solid #1C1C1C; }
+ .new-landing-page .flex-order-xs-1 {
+ order: 1 !important; }
+ .new-landing-page .flex-order-xs-2 {
+ order: 2 !important; }
+ .new-landing-page .layout-xs-column {
+ box-sizing: border-box;
+ display: flex; }
+ .new-landing-page .heading-over-video {
+ position: static;
+ bottom: 0px;
+ width: 100%;
+ margin: 0px;
+ border-bottom: 0px;
+ border-left: 0px;
+ border-right: 0px;
+ max-width: 100%; }
+ .new-landing-page .video-section .heading-over-video {
+ position: absolute; }
+ .new-landing-page .heading-over-video.right-card {
+ right: 0px; }
+ .new-landing-page .header-section {
+ margin: 0px; }
+ .new-landing-page .banner-section {
+ position: static;
+ display: flex;
+ flex-direction: column-reverse;
+ height: auto;
+ border: none; }
+ .new-landing-page .p-header {
+ font-size: 19px; }
+ .new-landing-page .tile-image.tile {
+ min-height: 500px; }
+ .new-landing-page .text-left {
+ text-align: center;
+ justify-content: center; }
+ .new-landing-page .banner-section {
+ height: 100%; }
+ .new-landing-page .btn-justify-left {
+ margin-left: 8px; }
+ .new-landing-page .card-text {
+ text-align: center;
+ border-left: 3px solid #1C1C1C;
+ border-right: 3px solid #1C1C1C;
+ align-self: stretch; }
+ .new-landing-page .video-section .card-text {
+ border-left: none;
+ border-right: none; }
+ .new-landing-page .video-section {
+ border-bottom: 0px solid #1C1C1C; }
+ .new-landing-page .pd-60 {
+ padding: 0px; }
+ .new-landing-page .border-black {
+ border-top: 0px solid #1C1C1C; }
+ .new-landing-page .mb-10 {
+ margin-bottom: 0px; }
+ .new-landing-page .banner-section {
+ border-bottom: 0px solid #1C1C1C;
+ border-top: 0px solid #1C1C1C; }
+ .new-landing-page #learn-more {
+ margin: 0px auto 0px auto; }
+ .new-landing-page #open-platform {
+ flex-direction: column-reverse;
+ border-top: 3px solid #1C1C1C;
+ border-bottom: 3px solid #1C1C1C; }
+ .new-landing-page #open-platform .flex-xs-100 {
+ flex-direction: column;
+ border-bottom: 0px solid #1C1C1C; }
+ .new-landing-page .mr-30 {
+ margin-right: 0px; }
+ .new-landing-page .new-landing-page {
+ padding-bottom: 0px;
+ background-size: 0px; }
+ .new-landing-page .banner-section {
+ background: none; } }
+ @media only screen and (max-width: 750px) {
+ .new-landing-page h1 {
+ font-size: 30px; }
+ .new-landing-page h2 {
+ font-size: 25px; }
+ .new-landing-page .text-funding {
+ text-align: center; } }
+ @media only screen and (max-width: 600px) {
+ .new-landing-page h1 {
+ font-size: 28px;
+ line-height: 33px; }
+ .new-landing-page h2 {
+ font-size: 28px; }
+ .new-landing-page p {
+ font-size: 19px; }
+ .new-landing-page .p-header {
+ font-size: 19px; }
+ .new-landing-page .p-60 {
+ padding: 10px 10px !important; }
+ .new-landing-page .my-20 {
+ margin-top: 10px !important;
+ margin-bottom: 10px !important; }
+ .new-landing-page .tile-image.tile {
+ min-height: 300px; }
+ .new-landing-page .tile {
+ padding: 50px 30px; } }
+
+/**
+ * NAVBAR
+ */
+.navbar_container a:hover, .navbar_container a:active {
+ color: white; }
+
+.navbar_container .md-button.navbar_highlight_button {
+ border: 2px solid #FFC100;
+ color: #FFC100;
+ border-radius: 50px;
+ padding-top: .01rem;
+ padding-bottom: .01rem;
+ transition: color 0.4s cubic-bezier(0.25, 0.8, 0.25, 1), background-color 0.4s cubic-bezier(0.25, 0.8, 0.25, 1); }
+ .navbar_container .md-button.navbar_highlight_button:hover, .navbar_container .md-button.navbar_highlight_button:active {
+ background-color: #FFC100 !important;
+ color: #1C1C1C !important; }
+
+.navbar_container .logo_link {
+ text-align: center; }
+ .navbar_container .logo_link .logo_icon {
+ height: 32px;
+ width: 32px; }
+
+.navbar_container .nav_icon {
+ margin-right: 4px;
+ height: 14px !important;
+ width: 14px !important; }
+ .navbar_container .nav_icon svg {
+ fill: #FFC100; }
+
+.navbar_container .navbar_signup_button .md-button {
+ padding: 2px 0px;
+ border: 2px solid #FFC100;
+ color: #FFC100;
+ border-radius: 50px;
+ width: 79px; }
+
+.navbar_container .navbar_signup_button .md-button:hover {
+ background-color: #FFC100;
+ color: #1C1C1C; }
+
+.navbar_avatar_icon {
+ width: 38px;
+ height: 38px;
+ border-radius: 19px; }
+
+.md-open-menu-container.md-active {
+ margin-top: 50px !important; }
+
+#search input {
+ background: url(../assets/images/search_icon_black.svg);
+ background-repeat: no-repeat;
+ background-position: 10px 12px;
+ background-size: 18px;
+ padding: 0px 35px; }
+
+.search_results {
+ display: flex;
+ align-items: center; }
+ .search_results .result_icon {
+ flex: 0 0 18px;
+ margin-right: 12px;
+ height: 18px; }
+ .search_results .result_name {
+ margin-right: 10px; }
+ .search_results .result_location {
+ color: #0019FF; }
+
+/*
+ CONTAINERS USED ON SHOW KIT STATE
+ */
+.over_map {
+ min-height: 120px; }
+
+.kit_fixed {
+ z-index: 2;
+ position: relative; }
+
+.kit_data {
+ width: 100%;
+ margin-top: 376px; }
+ .kit_data section.overlay {
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ background-color: white;
+ z-index: 3;
+ color: #1C1C1C;
+ text-align: center;
+ font-size: 20px;
+ font-weight: lighter; }
+ .kit_data section.overlay h1 {
+ color: #1C1C1C;
+ font-size: 1.8em;
+ margin: 0; }
+ .kit_data section.overlay p {
+ color: #D2D3D5;
+ margin: 0.5em; }
+ .kit_data .hint {
+ background-color: white;
+ text-align: center;
+ color: #D2D3D5;
+ margin-top: 0px;
+ z-index: 10;
+ position: absolute;
+ height: 100%;
+ width: 100%; }
+ .kit_data .hint > p {
+ color: #6E6E6E; }
+ .kit_data .hint .md-button {
+ margin-top: 20px; }
+
+.kit_data .shadow, .kitTags__section .shadow {
+ top: 0px;
+ height: 1px;
+ background-color: rgba(0, 0, 0, 0.1);
+ position: relative;
+ box-shadow: 0px -1px 6px 0 rgba(0, 0, 0, 0.65);
+ z-index: 2; }
+
+.kit_tags-select-header {
+ height: 48px;
+ display: flex; }
+
+.kit_tags-header-searchbox {
+ border: none;
+ border-bottom: 1px solid #ccc;
+ padding-left: 12px;
+ height: 100%;
+ width: 100%; }
+
+/*
+=========================================
+KIT COMMENTS MODULE
+=========================================
+*/
+.kit_comments {
+ padding-bottom: 100px; }
+
+/*
+ CHART COMPONENT
+*/
+.chart_container {
+ /*min-width: 340px;*/
+ min-height: 310px;
+ /*width: 100%;*/
+ /*height: 90%;*/
+ position: relative; }
+ .chart_container .chart_line {
+ fill: none;
+ stroke-width: 2px; }
+ .chart_container .chart_area {
+ opacity: 0.1; }
+ .chart_container .axis path, .chart_container .axis line {
+ fill: none;
+ stroke: #6E6E6E;
+ stroke-width: 1;
+ shape-rendering: crispEdges;
+ display: none; }
+ .chart_container .axis {
+ font-size: 10px; }
+ .chart_container .axis.y_left {
+ fill: #D2D3D5; }
+ .chart_container .axis.y_right {
+ fill: #D2D3D5; }
+ .chart_container .axis.x {
+ fill: #D2D3D5; }
+ .chart_container .grid .tick {
+ stroke: #D2D3D5;
+ stroke-opacity: 0.6;
+ shape-rendering: crispEdges; }
+ .chart_container .grid path {
+ stroke-width: 0; }
+ .chart_container .overlay {
+ fill: none;
+ pointer-events: all; }
+ .chart_container .focus circle {
+ fill: #03252d;
+ stroke-width: 2px; }
+ .chart_container .text_hover_container {
+ border: 1px solid #6E6E6E; }
+ .chart_container .popup_value {
+ fill: #1C1C1C;
+ color: #1C1C1C;
+ font-size: 18px;
+ font-weight: bold; }
+ .chart_container .popup_date {
+ fill: #6E6E6E;
+ font-size: 10px;
+ font-weight: bold; }
+ .chart_container .popup_unit {
+ fill: #D2D3D5;
+ margin-left: 10px; }
+
+.stickNav {
+ position: fixed;
+ top: 0px; }
+
+.overlay-kitinfo {
+ background: white;
+ height: 100%;
+ width: 100%;
+ position: absolute;
+ z-index: 999; }
+
+/*
+ KIT MENU MODULE
+ */
+section .kit_menu {
+ width: 100%;
+ z-index: 5;
+ background-color: #ECECEE;
+ position: relative;
+ /*Show raw Toggle*/
+ /* Rounded slider */ }
+ section .kit_menu .container {
+ width: 100%;
+ height: 33px; }
+ section .kit_menu .kit_time {
+ position: relative;
+ text-align: center;
+ color: #6E6E6E;
+ letter-spacing: 0.5px;
+ font-size: 14px;
+ text-transform: uppercase; }
+ section .kit_menu .kit_user {
+ margin-left: 20px;
+ line-height: 30px;
+ color: #6E6E6E;
+ font-weight: bold;
+ font-size: 13px; }
+ section .kit_menu .kit_user img {
+ border-radius: 50%;
+ float: left;
+ width: 18px;
+ height: 18px;
+ margin-right: 8px;
+ margin-top: 5px; }
+ section .kit_menu .kit_user a {
+ color: #6E6E6E;
+ text-decoration: none; }
+ section .kit_menu .kit_name {
+ margin-left: 30px;
+ line-height: 30px;
+ font-size: 13px;
+ font-weight: bold;
+ color: #6E6E6E; }
+ section .kit_menu .kit_name md-icon {
+ float: left;
+ width: 14px;
+ height: 14px;
+ margin-right: 8px;
+ margin-top: 4px; }
+ section .kit_menu .kit_name md-icon svg {
+ fill: #FFC100; }
+ section .kit_menu .kit_name span {
+ vertical-align: middle; }
+ section .kit_menu .kit_battery {
+ margin-left: 30px;
+ line-height: 30px;
+ color: #6E6E6E;
+ font-size: 12px;
+ font-weight: bold; }
+ section .kit_menu .kit_battery md-icon {
+ margin-bottom: 2px;
+ width: 16px;
+ margin-right: 8px; }
+ section .kit_menu .kit_battery md-icon svg {
+ fill: #6E6E6E; }
+ section .kit_menu .kit-show-raw {
+ margin-right: 6px; }
+ section .kit_menu .kit-raw-toggle {
+ vertical-align: middle; }
+ section .kit_menu .switch {
+ vertical-align: middle;
+ position: relative;
+ display: inline-block;
+ width: 30px;
+ height: 15px; }
+ section .kit_menu .switch input {
+ opacity: 0;
+ width: 0;
+ height: 0; }
+ section .kit_menu .slider {
+ position: absolute;
+ vertical-align: middle;
+ cursor: pointer;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 0;
+ background-color: #ccc;
+ transition: .4s; }
+ section .kit_menu .slider:before {
+ position: absolute;
+ content: "";
+ height: 11px;
+ width: 11px;
+ left: 3px;
+ bottom: 2px;
+ background-color: white;
+ transition: .4s; }
+ section .kit_menu input:checked + .slider {
+ background-color: #FFC100; }
+ section .kit_menu input:focus + .slider {
+ box-shadow: 0 0 1px #2196F3; }
+ section .kit_menu input:checked + .slider:before {
+ -webkit-transform: translateX(14px);
+ transform: translateX(14px); }
+ section .kit_menu .slider.round {
+ border-radius: 34px; }
+ section .kit_menu .slider.round:before {
+ border-radius: 50%; }
+ section .kit_menu .kit-show-raw-text {
+ vertical-align: middle;
+ color: #6E6E6E;
+ font-size: 12px;
+ font-weight: bold; }
+ section .kit_menu .kit_navbar {
+ margin-right: 20px; }
+ section .kit_menu .kit_navbar md-icon {
+ width: 14px; }
+ section .kit_menu .kit_navbar md-icon svg {
+ fill: #6E6E6E; }
+ section .kit_menu .kit_navbar .md-button {
+ margin: 0; }
+ section .kit_menu .kit_navbar a.md-button:hover, section .kit_menu .kit_navbar .button_active {
+ border-bottom: 2px solid #1C1C1C; }
+
+.kit_overview {
+ width: 100%;
+ height: 96px;
+ z-index: 2;
+ background-color: #F9F9FB;
+ position: relative; }
+ .kit_overview .button_scroll {
+ margin: 0px;
+ background: #ccc; }
+ .kit_overview .button_scroll_left {
+ left: 0; }
+ .kit_overview .button_scroll_right {
+ right: 0; }
+ .kit_overview .sensors_container {
+ width: 90%;
+ height: 96px;
+ margin: 0 auto;
+ overflow-x: hidden;
+ overflow-y: hidden; }
+ @media (pointer: coarse) {
+ .kit_overview .sensors_container {
+ overflow-x: auto; } }
+ .kit_overview .sensor_container {
+ position: relative;
+ color: #6E6E6E;
+ display: inline-table;
+ cursor: pointer;
+ text-align: center;
+ outline: none; }
+ .kit_overview .sensor_container p {
+ margin: 0px;
+ padding-bottom: 15px;
+ font-size: 0.8em;
+ font-weight: normal; }
+ .kit_overview .sensor_container .clear {
+ clear: both; }
+ .kit_overview .sensor_container .sensor_right {
+ position: absolute;
+ right: 15px;
+ bottom: 38px;
+ max-width: 30px; }
+ .kit_overview .sensor_container .sensor_right .sensor_unit {
+ margin-top: 8px;
+ font-size: 14px; }
+ .kit_overview .sensor_container .sensor_right .sensor_arrow {
+ margin-top: -15px;
+ width: 14px;
+ height: 7px; }
+ .kit_overview .sensor_container .sensor_right .sensor_arrow.arrow_up svg {
+ fill: #00E597; }
+ .kit_overview .sensor_container .sensor_right .sensor_arrow.arrow_down svg {
+ fill: #FF3D4C; }
+ .kit_overview .sensor_container .sensor_right .sensor_arrow.equal svg {
+ fill: #55C4F5; }
+ .kit_overview .sensor_container .sensor_value {
+ margin-top: 10px;
+ padding-left: 50px;
+ padding-right: 50px; }
+ .kit_overview .sensor_container .sensor_value_null {
+ opacity: 0.15; }
+ .kit_overview .sensor_container .sensor_icon {
+ width: 22px;
+ height: 22px;
+ position: absolute;
+ bottom: 45px;
+ left: 25px; }
+ .kit_overview .sensor_container .sensor_icon svg {
+ fill: #6E6E6E; }
+ .kit_overview .sensor_container:hover {
+ opacity: 0.8; }
+ .kit_overview .sensor_container.selected {
+ color: #1C1C1C; }
+ .kit_overview .sensor_container.selected .sensor_icon svg {
+ fill: #1C1C1C; }
+
+/**
+ * KIT DETAILS
+ *
+ */
+.kit_details_notAuth_title {
+ margin: 0; }
+
+.kit_details_notAuth_subheader {
+ color: #6E6E6E; }
+
+.kit_details_notAuth_button {
+ float: right; }
+
+.kit_detailed {
+ /*height: 1000px;*/
+ z-index: 2;
+ background-color: white;
+ position: relative; }
+ .kit_detailed p.description {
+ line-height: inherit;
+ font-weight: lighter;
+ width: 75%; }
+ .kit_detailed .kit_timeOpts {
+ margin: 0;
+ font-size: 1.3em; }
+ .kit_detailed .kit_chart {
+ min-height: 340px;
+ background-color: white;
+ position: relative; }
+ .kit_detailed .kit_chart .hint {
+ background-color: #1C1C1C;
+ opacity: 0.6; }
+ .kit_detailed .kit_chart .container {
+ padding: 37px;
+ height: 100%; }
+ .kit_detailed .kit_chart .container .kit_chart_left {
+ color: #1C1C1C; }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_select .sensor_icon_selected {
+ float: left;
+ margin-right: 6px;
+ padding-top: 5px; }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_select .sensor_icon_selected svg {
+ fill: #1C1C1C; }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_select md-select {
+ margin: 0; }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_select md-select .md-select-value .md-text .select_image {
+ display: none; }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_description {
+ /*//color: $background_blue_light;*/
+ /*//height: 80px;*/
+ /*//margin-top: 0px; //30;*/
+ /*//position: relative;*/ }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_description h6 {
+ margin: 0;
+ margin-bottom: 3px;
+ font-size: 1em; }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_description small {
+ line-height: 1em; }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_description a {
+ margin-left: 10px;
+ text-decoration: none;
+ color: #6E6E6E; }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_description .sensor_description_full {
+ position: absolute;
+ top: -50px;
+ margin-top: 10px;
+ display: none;
+ overflow: visible;
+ z-index: 200;
+ background-color: rgba(255, 255, 255, 0.8);
+ color: #1C1C1C;
+ border-radius: 2px;
+ line-height: 1.5em;
+ padding: 10px;
+ box-shadow: 0 1px 5px rgba(0, 0, 0, 0.65); }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_data .sensor_value {
+ font-size: 2em;
+ font-weight: bold; }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_data .sensor_unit {
+ font-size: .8em;
+ position: relative;
+ top: -8px; }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_data .sensor_data_icon {
+ width: 14px;
+ height: 14px;
+ margin-left: 8px; }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_data_description {
+ font-size: 12px;
+ display: none;
+ position: absolute;
+ -webkit-transform: translate(5px, 5px);
+ transform: translate(5px, 5px);
+ background-color: rgba(255, 255, 255, 0.8);
+ color: #1C1C1C;
+ border-radius: 2px;
+ line-height: 1.5em;
+ padding: 4px 6px;
+ box-shadow: 0 1px 5px rgba(0, 0, 0, 0.65);
+ max-width: 175px; }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_compare {
+ font-size: .8em;
+ padding-top: 30px;
+ margin-top: 10px; }
+ .kit_detailed .kit_chart .container .kit_chart_right {
+ /*//display: inline-block;*/
+ /*//height: 100%;*/ }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation {
+ height: 10%;
+ font-size: 11px; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation > div {
+ margin-left: 20px; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation input {
+ margin-left: 10px;
+ padding-left: 10px;
+ font-size: 12px; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .kit_time md-icon {
+ width: 11px;
+ height: 11px;
+ margin-right: 8px; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .picker_container {
+ margin-left: 20px; }
+ @media only screen and (max-device-width: 1050px) {
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_navigation {
+ justify-content: space-around !important;
+ margin: 5px !important; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .picker_container {
+ margin: 0 5px 0 5px !important; } }
+ @media only screen and (max-device-width: 960px) {
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .picker_container {
+ display: flex !important;
+ align-content: middle !important;
+ margin: 10px 0 0 0 !important;
+ width: 100% !important;
+ justify-content: space-between !important;
+ align-items: center !important; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .word_picker {
+ display: inline-block !important;
+ width: 100% !important; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move {
+ display: flex !important;
+ width: 100% !important;
+ justify-content: space-between !important;
+ margin-top: 15px !important; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_button {
+ width: 100% !important; } }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_button {
+ margin: 0px -2px 0 -2px;
+ width: 40px;
+ height: 28px;
+ display: inline-flex;
+ border: 1px solid #6E6E6E;
+ background-color: #F9F9FB; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_right {
+ border-top-left-radius: 0px;
+ border-bottom-left-radius: 0px; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_right svg {
+ fill: #6E6E6E; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_right:hover {
+ background-color: #6E6E6E; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_right:hover svg {
+ fill: #F9F9FB; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_left svg {
+ fill: #6E6E6E; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_left:hover {
+ background-color: #6E6E6E; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move .chart_move_left:hover svg {
+ fill: #F9F9FB; }
+ .kit_detailed .kit_chart .container .kit_chart_right .chart_navigation .chart_move md-icon {
+ width: 5.5px;
+ height: 9px;
+ position: relative;
+ top: 0px; }
+ .kit_detailed .kit_details {
+ color: #1C1C1C; }
+ .kit_detailed .kit_details_content svg {
+ fill: #1C1C1C; }
+ .kit_detailed .kit_details_content .kit_details_labels {
+ margin-top: 30px; }
+ .kit_detailed .kit_details_name {
+ margin: 5px 0 20px 0; }
+ .kit_detailed .kit_details_type > span {
+ margin-right: 15px; }
+ .kit_detailed .kit_details_type_location svg {
+ fill: #6E6E6E; }
+ .kit_detailed .kit_details_version span {
+ color: #6E6E6E;
+ font-weight: bold; }
+ .kit_detailed .kit_details_id span {
+ color: #6E6E6E;
+ font-weight: bold; }
+ .kit_detailed .kit_details_description {
+ padding-top: 4px; }
+ .kit_detailed .kit_details_description_title {
+ font-size: 16px;
+ margin-bottom: 15px; }
+ .kit_detailed .kit_details_description_content {
+ font-size: 12px; }
+ .kit_detailed .kit_details_manage_buttons {
+ margin-bottom: 32px; }
+ .kit_detailed .kit_details_button_delete {
+ color: #FF3D4C; }
+ .kit_detailed .kit_details_key {
+ float: left;
+ height: 26px; }
+ .kit_detailed .md-button.kit_details_key_refreshButton:hover, .kit_detailed .md-button.kit_details_key_refreshButton:focus {
+ background-color: #DBDBDB; }
+ .kit_detailed .kit_detailed_icon_content {
+ width: 14px;
+ height: 14px;
+ margin-right: 4px;
+ position: relative;
+ top: -1px; }
+ .kit_detailed .kit_detailed_icon_content svg {
+ fill: #6E6E6E; }
+ .kit_detailed .kit_detailed_icon_title {
+ width: 18px;
+ height: 18px;
+ margin-right: 8px;
+ position: relative;
+ top: -2px; }
+ .kit_detailed .kit_detailed_title_container {
+ padding: 20px 0; }
+ .kit_detailed .kit_detailed_title {
+ font-size: 16px;
+ display: inline; }
+ .kit_detailed .kit_detailed_content_container {
+ margin: 62px 37px 0px; }
+ .kit_detailed .kit_owner {
+ color: #1C1C1C;
+ background-color: #F5F5F5;
+ padding-bottom: 40px; }
+ .kit_detailed .kit_owner .container {
+ width: 95%;
+ margin: 0 auto; }
+ .kit_detailed .kit_owner .kit_owner_usernameLink {
+ text-decoration: none; }
+ .kit_detailed .kit_owner .kit_owner_usernameText {
+ color: #1C1C1C;
+ margin: 50px 0 20px 0; }
+ .kit_detailed .kit_owner .kit_owner_location {
+ color: #6E6E6E;
+ font-weight: bold; }
+ .kit_detailed .kit_owner .kit_owner_url a {
+ text-decoration: none;
+ color: #1C1C1C; }
+ .kit_detailed .kit_owner .kit_owner_kits_title {
+ margin-bottom: 50px; }
+ .kit_detailed .kit_owner .kit_owner_kits_list {
+ border: 1px solid #EAECF1;
+ height: 82px;
+ background-color: white; }
+ .kit_detailed .kit_owner .kit_owner_kits_list_avatar {
+ float: left;
+ width: 44px;
+ height: 44px;
+ margin-right: 3%;
+ margin-top: 9px; }
+ .kit_detailed .kit_owner .kit_owner_kits_list_content {
+ float: left;
+ margin-top: 5px; }
+ .kit_detailed .kit_owner .kit_owner_kits_list_data {
+ font-size: 16px; }
+ .kit_detailed .kit_owner .kit_owner_kits_list_data span {
+ margin-right: 15px; }
+ .kit_detailed .kit_owner .kit_owner_kits_list_state {
+ display: inline;
+ font-size: 12px;
+ border-radius: 20px;
+ padding: 6px 11px;
+ margin: 15px; }
+ .kit_detailed .kit_owner .kit_owner_kits_list_never_published {
+ background-color: #FFC100; }
+ .kit_detailed .kit_owner .kit_owner_kits_list_not_configured {
+ background-color: #FF3D4C; }
+ .kit_detailed .kit_owner .kit_owner_kits_list_config {
+ float: right;
+ margin-top: 13px; }
+ .kit_detailed .kit_owner .kit_owner_kits_list_config_icon {
+ width: 22px;
+ height: 22px; }
+
+.sensor_value {
+ font-size: 34px;
+ font-family: Kanit;
+ font-weight: 700; }
+
+/**/
+/* [!] Leave animation (ng-leave ) are disabled
+because they conflict with enter animations */
+.sensor_animation.ng-enter {
+ transition: 1s ease-out all; }
+
+.sensor_animation.ng-enter {
+ -webkit-transform: scale(1.05);
+ transform: scale(1.05); }
+
+.sensor_animation.ng-enter-active {
+ -webkit-transform: scale(1);
+ transform: scale(1); }
+
+/* .sensor_animation.ng-leave{
+ transition:1s ease all;
+}
+.sensor_animation.ng-leave{
+ opacity: 0;
+}
+.sensor_animation.ng-leave-active{
+ opacity: 1;
+} */
+/* Time animation currently not in use */
+.time_animation.ng-enter {
+ transition: 1s ease-out all; }
+
+.time_animation.ng-enter {
+ -webkit-transform: scale(1.05);
+ transform: scale(1.05); }
+
+.time_animation.ng-enter-active {
+ -webkit-transform: scale(1);
+ transform: scale(1); }
+
+/* .time_animation.ng-leave {
+ transition:1s ease all;
+}
+.time_animation.ng-leave {
+ opacity: 0;
+}
+.time_animation.ng-leave-active {
+ opacity: 1;
+} */
+.bat_animation.ng-enter {
+ transition: 1s ease-out all; }
+
+.bat_animation.ng-enter {
+ -webkit-transform: scale(1.05);
+ transform: scale(1.05); }
+
+.bat_animation.ng-enter-active {
+ -webkit-transform: scale(1);
+ transform: scale(1); }
+
+/* .bat_animation.ng-leave {
+ transition:2s ease all;
+}
+.bat_animation.ng-leave {
+ opacity: 0;
+}
+.bat_animation.ng-leave-active {
+ opacity: 1;
+} */
+/*
+ MODAL STORE
+*/
+.store_itemDescription {
+ margin-left: 20px; }
+ .store_itemDescription ul {
+ list-style-image: url("../assets/images/check_circle.svg");
+ list-style-position: inherit;
+ font-size: .8em;
+ line-height: 1.8em; }
+
+.csv_file_item {
+ display: flex;
+ align-items: center; }
+ .csv_file_item md-checkbox {
+ margin-bottom: 0px; }
+ .csv_file_item span {
+ margin-right: 20px; }
+
+.list-shadow {
+ box-shadow: 0px 1px 3px 0px rgba(0, 0, 0, 0.2); }
+ .list-shadow md-list-item {
+ border-bottom: solid 1px #ECECEE; }
+ .list-shadow md-list-item:last-child {
+ border-bottom: none; }
+
+.label-grey {
+ background-color: #D2D3D5; }
+
+.title-timeline {
+ font-size: 30px; }
+
+/* This is temporary */
+.upload-csv .container {
+ margin-right: auto;
+ margin-left: auto; }
+
+.upload-csv .csv_content {
+ margin-top: 240px;
+ min-height: 50vh; }
+
+@media (min-width: 768px) {
+ .upload-csv .container {
+ width: 750px; } }
+
+@media (min-width: 992px) {
+ .upload-csv .container {
+ width: 970px; } }
+
+@media (min-width: 1200px) {
+ .upload-csv .container {
+ width: 1170px; } }
+
+/**
+ * BACKDROP COMPONENT
+ */
+md-content.md-opaque {
+ background-color: rgba(30, 30, 30, 0.6); }
+
+md-content.md-mainBackdrop {
+ display: flex;
+ align-items: center;
+ justify-content: center;
+ width: 100vw;
+ height: 100vh; }
+ md-content.md-mainBackdrop .backdrop_icon {
+ width: 101px;
+ height: 101px; }
+
+md-content.md-stateChangeBackdrop {
+ z-index: 999;
+ position: absolute;
+ background-color: transparent;
+ width: 100vw;
+ height: 100vh; }
+
+.backdrop_icon svg {
+ overflow: visible;
+ fill: #6E6E6E; }
+
+.backdrop_icon .sc-circle-arrow {
+ -webkit-transform-origin: 50% 53.8%;
+ transform-origin: 50% 53.8%;
+ -webkit-animation: spin 3s linear infinite;
+ animation: spin 3s linear infinite; }
+
+@-webkit-keyframes spin {
+ 100% {
+ -webkit-transform: rotate(360deg); } }
+
+@keyframes spin {
+ 100% {
+ -webkit-transform: rotate(360deg);
+ transform: rotate(360deg); } }
+
+section .stickMenu {
+ position: absolute !important; }
+
+.stickNav {
+ position: fixed;
+ top: 0px;
+ border-bottom: 1px solid #666; }
+
+/**
+ * MOZILLA-SPECIFIC CSS
+ */
+@-moz-document url-prefix() {
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_select .sensor_icon_selected {
+ transform: translateY(-8px); }
+ .kit_detailed .kit_chart .container .kit_chart_left .sensor_compare md-select {
+ transform: translateY(8px);
+ margin: 0; }
+ body .kit_detailed .kit_chart .container .kit_chart_left .sensor_data {
+ margin: 5% 0 4% 0; }
+ /* section.map {
+ z-index: 0;
+ } */ }
+
+/**
+ * INPUT CSS
+ */
+html {
+ font-family: 'Roboto Condensed';
+ font-size: 18px; }
+
+body {
+ background-color: rgba(0, 0, 0, 0);
+ color: #1C1C1C; }
+
+a.about {
+ cursor: pointer;
+ text-decoration: none;
+ color: #0019FF; }
+
+li.policy-toc {
+ padding: 3px; }
+ li.policy-toc a {
+ text-decoration: none; }
+
+ul#policy-toc {
+ padding-bottom: 0px !important;
+ padding-bottom: 0px !important; }
+
+.policy-toc#header {
+ list-style: none; }
+
+.icon_label {
+ width: 14px;
+ height: 14px;
+ margin-right: 4px;
+ position: relative;
+ top: -2px; }
+ .icon_label svg {
+ fill: #6E6E6E; }
+
+/**
+ * TYPOGRAPHY
+ */
+h1 {
+ font-size: 2.2em;
+ font-weight: normal; }
+
+h2 {
+ font-size: 1.7em;
+ font-weight: normal; }
+
+h3 {
+ font-size: 1.4em;
+ font-weight: normal; }
+
+h4 {
+ font-size: 1.2em;
+ font-weight: normal; }
+
+p {
+ line-height: 1.5; }
+
+td {
+ padding: 5px; }
+
+input {
+ font-family: 'Roboto Condensed'; }
+
+.label, .tag, .state {
+ text-transform: uppercase;
+ font-size: 12px;
+ line-height: 24px;
+ border-radius: 20px;
+ padding: 3px 8px;
+ margin-right: 8px; }
+ .label md-icon, .tag md-icon, .state md-icon {
+ margin-left: 4px;
+ height: 11px;
+ width: 11px;
+ margin-top: -3px;
+ cursor: pointer; }
+ .label[clickable], .tag[clickable], .state[clickable] {
+ cursor: pointer; }
+
+.label {
+ background-color: #FFD44D;
+ color: #1C1C1C; }
+ .label md-icon svg path {
+ fill: #D2D3D5; }
+
+.tag {
+ background-color: #FFECB3;
+ color: #1C1C1C; }
+ .tag md-icon svg path {
+ fill: #6E6E6E; }
+
+.leaflet-container .tag, .leaflet-container .label,
+.leaflet-container .state {
+ font-size: 1em; }
+
+.leaflet-top {
+ top: 93px; }
+
+section.info h1 {
+ font-size: 1.3em; }
+
+a {
+ color: #1C1C1C; }
+ a:hover, a:active {
+ text-decoration: none;
+ color: #0019FF; }
+
+a.footer-link {
+ color: #FFC100;
+ text-decoration: none;
+ transition: color 0.4s cubic-bezier(0.25, 0.8, 0.25, 1); }
+ a.footer-link:hover, a.footer-link:active {
+ text-decoration: none;
+ color: #0019FF; }
+
+/*
+ Overwrite global material design, allow icons to be smaller
+*/
+md-toolbar {
+ background-color: #1C1C1C !important;
+ z-index: 10;
+ color: white !important; }
+ md-toolbar md-icon {
+ fill: white !important; }
+
+md-icon {
+ min-width: unset !important; }
+
+md-select md-select-value {
+ border-bottom-color: #cccccc75 !important; }
+
+a.md-button h4, a.md-button .h4 {
+ color: #0019FF;
+ margin: 0; }
+
+a.warn,
+button.warn {
+ color: #FF3D4C; }
+
+.bold {
+ font-weight: bold; }
+
+div[api-key] {
+ display: flex; }
+ div[api-key] .api_key_number {
+ background-color: #F2F2F2;
+ word-break: break-all;
+ width: 420px; }
+ div[api-key] .api_key_refresh_button {
+ background-color: #DBDBDB;
+ margin: 0; }
+ div[api-key] .api_key_refresh_button md-icon {
+ padding: 10px;
+ height: 15px;
+ width: 15px; }
+ div[api-key] md-icon {
+ opacity: .5; }
+
+.info_overlay h2 {
+ color: white; }
+
+.info_overlay p {
+ margin: 0;
+ font-size: 20px;
+ font-weight: 300;
+ color: #D2D3D5; }
+
+.dark h1, .dark h2 {
+ color: white; }
+
+.dark h1, .dark h2 {
+ margin: 0; }
+
+.dark h3, .dark h4, .dark .h4, .dark a {
+ color: #D2D3D5; }
+
+/**
+ * ANGULAR MATERIAL OVERRIDING
+ */
+md-select-menu {
+ background-color: white; }
+
+md-select,
+md-select-menu {
+ text-transform: uppercase; }
+
+md-select:not([disabled]):focus .md-select-value {
+ border-bottom-color: #6B868D; }
+
+.md-button.md-primary.md-flat:not([disabled]):hover {
+ background-color: #0019FF;
+ color: white; }
+
+.md-button.md-primary.md-flat:not([disabled]):focus {
+ background-color: #0019FF;
+ color: white; }
+
+.md-button.fillwidth {
+ width: 100% !important; }
+
+md-input-container:not(.md-input-invalid).md-input-focused label {
+ color: #0019FF; }
+
+md-input-container:not(.md-input-invalid).md-input-focused .md-input {
+ border-color: rgba(0, 0, 0, 0.12); }
+
+/**
+ * ALERT COMPONENT
+ */
+.md-toast-content {
+ box-shadow: unset !important;
+ background-color: unset !important;
+ color: black !important;
+ min-width: 100%; }
+
+md-toast {
+ bottom: unset !important;
+ color: black;
+ opacity: 0.95;
+ padding: 0;
+ position: fixed;
+ top: 64px !important;
+ min-width: 100%;
+ z-index: 15; }
+ md-toast .alert_typeIcon {
+ width: 16px;
+ height: 16px;
+ margin-right: 16px; }
+ md-toast .alert_closeIcon {
+ width: 10.5px;
+ height: 10.5px;
+ margin: 0 auto; }
+ md-toast .alert_button {
+ padding: 0px 12px;
+ margin-left: 5px;
+ color: black !important;
+ background-color: #FFC100;
+ border-radius: 20px;
+ font-size: 12px; }
+
+md-toast.red {
+ background-color: #FF3D4C; }
+
+md-toast.green {
+ background-color: #4DBD4E; }
+
+md-toast.blue {
+ background-color: #0019FF; }
+
+md-toast.yellow {
+ background-color: #FFC100; }
+
+md-checkbox.md-checked .md-icon {
+ background-color: #0019FF; }
+
+md-checkbox .md-label {
+ pointer-events: auto; }
+
+md-checkbox .md-label span {
+ pointer-events: none;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none; }
+
+/**
+ * SPINNER/PROGRESS BAR COMPONENTS
+ */
+md-progress-linear.md-bar.green {
+ background-color: #00E597; }
+
+md-progress-linear.green .md-container {
+ background-color: rgba(0, 229, 151, 0); }
+
+md-progress-linear.green .md-bar {
+ background-color: #00E597; }
+
+md-progress-linear.md-bar {
+ background-color: #0019FF; }
+
+md-progress-linear.md-hue-3 .md-container {
+ background-color: rgba(0, 25, 255, 0.05); }
+
+md-progress-linear.md-hue-3 .md-bar {
+ background-color: #0019FF; }
+
+.kit_spinner {
+ z-index: 200;
+ position: absolute;
+ left: 0;
+ bottom: 6px; }
+
+.chart_spinner {
+ position: absolute;
+ top: 200px;
+ left: 57%; }
+
+/**
+ * COOKIES POLICY COMPONENT
+ */
+.cookies-policy_container {
+ background-color: #3c3c3c;
+ color: #FFC100;
+ position: fixed;
+ bottom: 0;
+ padding: 5px;
+ left: 0;
+ width: 100%;
+ z-index: 999;
+ text-align: center;
+ line-height: 4vh; }
+ .cookies-policy_container a {
+ color: rgba(255, 193, 0, 0.6); }
+
+.md-noDataBackdrop {
+ width: 100%;
+ height: 381px;
+ position: absolute;
+ background-color: white; }
+ .md-noDataBackdrop .block {
+ margin: 0 auto; }
+ .md-noDataBackdrop .title, .md-noDataBackdrop .message {
+ text-align: center; }
+ .md-noDataBackdrop .title {
+ color: #6E6E6E;
+ font-weight: 400;
+ margin-bottom: 0; }
+ .md-noDataBackdrop .message {
+ margin: 0;
+ font-size: 20px;
+ font-weight: 300; }
+ .md-noDataBackdrop a {
+ color: #0019FF;
+ text-decoration: none; }
+
+/**
+ * MAP
+ */
+.map_state {
+ position: fixed;
+ z-index: 1; }
+
+section.map {
+ background-color: white;
+ text-align: center;
+ width: 100%;
+ height: 100%;
+ color: black;
+ z-index: 1;
+ position: fixed;
+ top: 64px; }
+ @media all and (max-height: 1200px) and (min-height: 1000px) {
+ section.map {
+ height: 950px; } }
+ @media all and (max-height: 1000px) and (min-height: 800px) {
+ section.map {
+ height: 700px; } }
+ @media all and (max-height: 800px) and (min-height: 600px) {
+ section.map {
+ height: 550px; } }
+ @media all and (max-height: 600px) and (min-height: 400px) {
+ section.map {
+ height: 300px; } }
+ section.map h1 {
+ font-size: 3em; }
+
+/*
+ CORE CHIP COMPONENT
+ */
+.chips {
+ display: inline-block; }
+ .chips div.chips_row {
+ height: 40.5px; }
+ .chips .label, .chips .tag {
+ box-shadow: 0 1px 5px rgba(0, 0, 0, 0.65); }
+
+.chip {
+ padding: 0px 10px 0px 14px;
+ margin: auto 5px;
+ border-radius: 20px;
+ height: 30px;
+ text-transform: uppercase;
+ font-size: 13px;
+ line-height: 29px; }
+ .chip .chip_name {
+ margin-left: 3px;
+ margin-right: 7px; }
+ .chip .chip_icon {
+ border: 0;
+ background-color: inherit; }
+ .chip .chip_icon md-icon {
+ width: 8px;
+ height: 8px; }
+ .chip.map_filter {
+ background-color: white;
+ box-shadow: 0 1px 5px rgba(0, 0, 0, 0.65); }
+ .chip.kit_tag {
+ background-color: #D2D3D5;
+ float: left;
+ margin: 5px; }
+
+/*
+ MAP FILTERS
+*/
+.map_legend {
+ position: absolute;
+ top: 10px;
+ margin-left: 10px;
+ display: flex; }
+ .map_legend .chips {
+ align-self: flex-end;
+ padding-bottom: 5px;
+ padding-left: 5px;
+ display: flex; }
+ .map_legend .map_legend__filtersContainer {
+ display: flex;
+ flex-direction: column;
+ justify-content: space-between;
+ width: 124px;
+ height: 81px;
+ border-radius: 4px;
+ overflow: hidden;
+ box-shadow: 0 1px 5px rgba(0, 0, 0, 0.65); }
+ .map_legend .map_legend__filtersContainer svg, .map_legend .map_legend__filtersContainer p {
+ fill: #1C1C1C; }
+ .map_legend .map_legend__filtersContainer .map_legend__filtersRow {
+ cursor: pointer;
+ display: flex;
+ align-items: center;
+ padding-left: 5px;
+ background-color: #fff; }
+ .map_legend .map_legend__filtersContainer .map_legend__filtersRow:not(:last-child) {
+ border-bottom: 1px solid gray; }
+ .map_legend .map_legend__filtersContainer .map_legend__filtersRow .map_filter_button {
+ box-shadow: none; }
+ .map_legend .map_legend__filtersContainer .map_legend__filtersRow .map_filter_button p {
+ margin: 0; }
+
+.map_filter_button {
+ background-color: white;
+ margin-right: 15px;
+ width: 26px;
+ height: 26px;
+ box-shadow: 0 1px 5px rgba(0, 0, 0, 0.65); }
+ .map_filter_button md-icon {
+ width: 18px;
+ height: 12px; }
+
+.md-button.map_filter_button:hover, .md-button.map_filter_button:focus {
+ background-color: white; }
+
+/*
+=========================================
+KIT TAGS MODULE
+=========================================
+*/
+.kitTags__section {
+ margin-top: 376px; }
+ .kitTags__section .kitTags__listContainer {
+ background-color: #ECECEE;
+ padding: 20px; }
+ .kitTags__section .kitTags__container {
+ padding-bottom: 20px; }
+ .kitTags__section .kitTags__container .kitTags__textContainer {
+ display: flex;
+ justify-content: space-between;
+ margin-left: 20px;
+ margin-right: 20px; }
+ .kitTags__section .kitTags__container .kitTags__textContainer .kitTags__textElement {
+ flex-basis: 40%;
+ flex-grow: 0;
+ min-width: 300px; }
+
+/*
+ CORE FORM COMPONENT
+ */
+.form_container {
+ font-size: 16px;
+ position: relative;
+ border-radius: 0px 0px 3px 3px; }
+
+.form_contentContainer {
+ width: 92%;
+ margin: 0 auto; }
+
+.form_field {
+ position: relative; }
+
+.form_errors {
+ position: absolute;
+ bottom: 15%;
+ left: 5%;
+ color: red;
+ font-size: 0.7em;
+ line-height: 1; }
+
+.form_closeIcon {
+ float: right;
+ width: 14px;
+ height: 14px;
+ margin: 12px; }
+ .form_closeIcon md-icon {
+ width: 14px;
+ height: 14px;
+ position: absolute;
+ top: 0px;
+ left: 0px; }
+
+.md-primary.form_button {
+ color: white;
+ background-color: #0019FF;
+ height: 64px;
+ bottom: 0px;
+ left: 0px;
+ border-radius: 0px 0px 2px 2px; }
+
+.form_title {
+ color: #0019FF;
+ font-weight: normal; }
+
+.form_messageHeader {
+ font-size: 1.5em;
+ color: #6E6E6E;
+ font-weight: bold;
+ line-height: 0;
+ margin-top: 50px; }
+
+.form_messageSubheader {
+ font-size: 0.875em;
+ color: rgba(0, 98, 123, 0.54);
+ line-height: 0.5; }
+
+.form_messageDescription {
+ font-size: 12px;
+ margin-bottom: 0; }
+
+/*
+ CORE LARGE DEVICES FORM
+*/
+@media (min-width: 501px) {
+ .form_container {
+ width: 340px;
+ font-size: 16px; }
+ .form_messageContainer {
+ margin-bottom: 20px; }
+ .md-primary.form_button {
+ width: 340px;
+ display: block;
+ position: relative;
+ margin: 0 auto;
+ font-size: 1.5em; } }
+
+/*
+ CORE MOBILE DEVICES FORM
+*/
+@media (max-width: 500px) {
+ .form_container {
+ width: 80%;
+ font-size: 12px; }
+ .md-primary.form_button {
+ width: 100%;
+ position: absolute;
+ font-size: 1.2em; } }
+
+@media only screen and (min-device-width: 320px) and (max-device-width: 480px) and (-webkit-device-pixel-ratio: 2) and (device-aspect-ratio: 2 / 3) {
+ .recovery_container {
+ margin-top: 40px; }
+ .form_field {
+ padding: 5% 0; } }
+
+/*
+ RECOVERY FORM
+*/
+.recovery_container {
+ position: relative;
+ margin: 0px auto;
+ margin-bottom: 100px; }
+
+.formRecovery_field {
+ padding: 10% 0; }
+
+@media (min-width: 501px) {
+ .recovery_container {
+ margin-top: 80px; }
+ .formRecovery_errors {
+ bottom: 15%;
+ left: 5%; } }
+
+@media (max-width: 500px) {
+ .recovery_container {
+ margin-top: 20%; }
+ .formRecovery_errors {
+ bottom: 23%;
+ left: 5%; } }
+
+@media only screen and (min-device-width: 320px) and (max-device-width: 480px) and (-webkit-device-pixel-ratio: 2) and (device-aspect-ratio: 2 / 3) and (max-width: 500px) {
+ .formRecovery_field {
+ padding: 2% 0; }
+ .formRecovery_errors {
+ bottom: 29%; } }
+
+/*
+ RESET FORM
+*/
+.formReset_field {
+ padding: 7% 0; }
+
+@media (min-width: 501px) {
+ .formReset_errors {
+ bottom: 5%;
+ left: 5%; } }
+
+@media (max-width: 500px) {
+ .formReset_errors {
+ bottom: 18%;
+ left: 1%; } }
+
+@media only screen and (min-device-width: 320px) and (max-device-width: 480px) and (-webkit-device-pixel-ratio: 2) and (device-aspect-ratio: 2 / 3) and (max-width: 500px) {
+ .formReset_errors {
+ top: 65%; }
+ .formReset_field {
+ padding: 3% 0; } }
+
+/**
+ * MOVE DOWN EFFECT ON SCROLL
+ */
+.move_down {
+ -webkit-transform: translateY(32px);
+ transform: translateY(32px); }
+
+/*
+ MARKER AND POPUP MARKER
+*/
+.markerSmartCitizenNormal {
+ border: 1px solid #FFC100;
+ border-radius: 20px;
+ box-shadow: 0px 2px 4px 0px rgba(0, 0, 0, 0.3); }
+
+.markerSmartCitizenOnline {
+ border: 2px solid #FFC100;
+ border-radius: 20px;
+ box-shadow: 0px 2px 4px 0px rgba(0, 0, 0, 0.3);
+ background-color: rgba(255, 193, 0, 0.5); }
+
+.markerExperimentalNormal {
+ border: 2px solid #D2D3D5;
+ border-radius: 20px;
+ box-shadow: 0px 2px 4px 0px rgba(0, 0, 0, 0.3);
+ background-color: #D2D3D5; }
+
+/* @keyframes pulse {
+ 0% {
+ opacity: 0.2;
+ }
+ 50% {
+ opacity: 1;
+ }
+ 100% {
+ opacity: 0.2;
+ }
+}
+@-webkit-keyframes pulse {
+ 0% {
+ opacity: 0.2;
+ }
+ 50% {
+ opacity: 1;
+ }
+ 100% {
+ opacity: 0.2;
+ }
+}
+@-moz-keyframes pulse {
+ 0% {
+ opacity: 0.2;
+ }
+ 50% {
+ opacity: 1;
+ }
+ 100% {
+ opacity: 0.2;
+ }
+}
+@-o-keyframes pulse {
+ 0% {
+ opacity: 0.2;
+ }
+ 50% {
+ opacity: 1;
+ }
+ 100% {
+ opacity: 0.2;
+ }
+} */
+.markerSmartCitizenOffline {
+ border: 2px solid rgba(0, 0, 0, 0.2);
+ border-radius: 20px;
+ box-shadow: 0px 2px 4px 0px rgba(0, 0, 0, 0.3);
+ background-color: rgba(0, 0, 0, 0.2); }
+
+/* .marker_blink {
+ -webkit-animation: pulse 2s linear infinite;
+ -moz-animation: pulse 2s linear infinite;
+ -ms-animation: pulse 2s linear infinite;
+ animation: pulse 2s linear infinite;
+} */
+.popup {
+ width: 101%;
+ height: 135px;
+ color: #1C1C1C;
+ font-family: 'Roboto Condensed'; }
+
+.popup_top {
+ height: 72px;
+ border-radius: 2px 2px 0 0;
+ background-color: #ECECEE;
+ padding-top: 5px; }
+ .popup_top a {
+ color: #1C1C1C; }
+ .popup_top a:hover {
+ color: #6E6E6E; }
+ .popup_top .popup_name {
+ padding: 0 0 1px 8px;
+ font-size: 16px;
+ font-weight: bold; }
+ .popup_top .popup_type {
+ font-size: 14px;
+ line-height: 1em;
+ padding: 1px 0 3px 8px; }
+ .popup_top .popup_time {
+ font-size: 11px;
+ padding: 2px 0 0 8px; }
+ .popup_top .popup_time md-icon {
+ width: 10px;
+ height: 10px;
+ margin-right: 5px; }
+
+.popup_bottom {
+ height: 68px; }
+ .popup_bottom .popup_location {
+ padding: 5px 0 0 8px;
+ font-size: 14px; }
+ .popup_bottom .popup_location md-icon {
+ width: 8.4px;
+ height: 12px;
+ margin-right: 4px; }
+ .popup_bottom .popup_labels {
+ padding: 8px 0 0 8px;
+ font-size: 12px; }
+ .popup_bottom .popup_labels tag, .popup_bottom .popup_labels span {
+ margin-bottom: 8px;
+ float: left; }
+
+.popup_icon svg:nth-child(2) {
+ width: 0;
+ height: 0; }
+
+/* Kit type colors kitUtils.service.js :: classify()*/
+.sck {
+ background-color: #FFC100; }
+
+.unknown {
+ background-color: grey; }
+
+/**
+ * OVERRIDE LEAFLET
+ */
+.leaflet-left {
+ left: unset !important;
+ right: 11px;
+ top: 15px !important; }
+
+.leaflet-popup-content-wrapper {
+ border-radius: 4px;
+ width: auto !important;
+ overflow: hidden; }
+
+.leaflet-popup-content {
+ margin: 0;
+ min-width: 245px;
+ position: relative;
+ top: -1px;
+ left: -1px; }
+ .leaflet-popup-content p {
+ margin: 0; }
+
+.leaflet-popup-close-button {
+ display: none; }
+
+.leaflet-overlay-pane .leaflet-zoom-animated path {
+ stroke: grey; }
+
+.leaflet-top.leaflet-left {
+ z-index: 1; }
+
+.leaflet-bottom.leaflet-right {
+ z-index: 1; }
+
+.marker-cluster {
+ box-shadow: 0px 2px 4px 0px rgba(0, 0, 0, 0.2); }
+ .marker-cluster div {
+ font-family: 'Roboto Condensed';
+ font-size: 14px; }
+
+.marker-cluster-small, .marker-cluster-medium, .marker-cluster-large {
+ border-radius: 50%;
+ border: 2px solid rgba(255, 193, 0, 0.8);
+ background-color: rgba(255, 193, 0, 0.3); }
+ .marker-cluster-small div, .marker-cluster-medium div, .marker-cluster-large div {
+ background-color: rgba(0, 255, 168, 0); }
+
+/*
+ PICKER COMPONENT
+*/
+.picker {
+ color: #1C1C1C; }
+
+.picker__weekday {
+ color: #6E6E6E; }
+
+.picker__nav--prev::before {
+ border-right: 0.75em solid #6E6E6E; }
+
+.picker__nav--next::before {
+ border-left: 0.75em solid #6E6E6E; }
+
+.picker__button--today::before {
+ border-top: 0.66em solid #0019FF; }
+
+.picker__holder.picker_container {
+ background: none;
+ transition: none; }
+
+.picker__button--clear::before {
+ border-top: 3px solid #FF3D4C; }
+
+.picker__button--close {
+ color: #1C1C1C; }
+
+.picker__button--today, .picker__button--clear {
+ color: #1C1C1C; }
+
+.picker__day--highlighted, .picker__day--selected, .picker__day--selected:hover, .picker--focused .picker__day--selected {
+ background: #FFC100;
+ background-color: #FFC100;
+ border-color: #FFC100;
+ color: #1C1C1C; }
+
+.picker__day--infocus:hover, .picker__nav--prev:hover, .picker__nav--next:hover, .picker__button--today:hover, .picker__button--clear:hover, .picker__button--close:hover {
+ background: #FFECB3;
+ border-color: #FFECB3; }
+
+.date_picker {
+ height: 24px;
+ border-radius: 2px;
+ border: 1px solid #6E6E6E;
+ color: #6E6E6E;
+ background-color: transparent; }
+
+.select_image {
+ margin-right: 10px; }
+
+/**
+ * DROPDOWN MENU
+ * Override style for dropdown menu from angular-dropdowns dependency
+ */
+.wrap-dd-menu .dropdown li a {
+ color: #6E6E6E; }
+
+.kit_menu .wrap-dd-menu {
+ width: 30px;
+ display: inline-table;
+ padding: 0 0; }
+
+.kit_menu .wrap-dd-menu .dropdown {
+ top: 70%;
+ line-height: 1em;
+ left: -25px;
+ right: -25px; }
+ .kit_menu .wrap-dd-menu .dropdown li a {
+ text-align: center;
+ border-bottom: 0px; }
+
+.kitList_parent .wrap-dd-menu {
+ width: 50px;
+ display: inline-block; }
+
+.kitList_parent .wrap-dd-menu .dropdown {
+ top: 70%;
+ line-height: 1em;
+ left: -25px;
+ right: -25px;
+ border-radius: 0px; }
+ .kitList_parent .wrap-dd-menu .dropdown li a {
+ text-align: center;
+ border-bottom: 0px; }
+
+.kit_owner_dropdown_button {
+ display: block;
+ margin: 0 auto; }
+
+/**
+ * KIT LIST
+ *
+ */
+.kitList {
+ margin: 0px;
+ color: #D2D3D5;
+ border-top: 1px solid #EAECF1;
+ border-right: 1px solid #EAECF1;
+ border-left: 1px solid #EAECF1;
+ background-color: white;
+ border-radius: 0 !important;
+ text-transform: none;
+ padding: 15px; }
+ .kitList h1, .kitList h2, .kitList h3, .kitList h4 {
+ line-height: inherit; }
+ .kitList p {
+ line-height: 0.4em; }
+ .kitList .wrap-dd-menu {
+ width: 50px;
+ display: inline-block; }
+ .kitList .wrap-dd-menu .dropdown {
+ top: 70%;
+ line-height: 1em;
+ left: -25px;
+ right: -25px;
+ border-radius: 0px; }
+ .kitList .wrap-dd-menu .dropdown li a {
+ text-align: center;
+ border-bottom: 0px; }
+ .kitList .chips {
+ position: absolute;
+ right: 130px; }
+
+.kitList_avatar {
+ width: 44px;
+ height: 44px;
+ border-radius: 22px; }
+
+.kitList_name {
+ margin-bottom: 5px;
+ font-size: 18px;
+ color: #0019FF; }
+
+.kitList_right {
+ position: absolute;
+ right: 0; }
+ .kitList_right .chip .chip_name {
+ text-transform: uppercase; }
+
+.kitList_state {
+ color: white;
+ text-transform: uppercase; }
+
+.kitList_state_has_published {
+ color: #00E597; }
+
+.kitList_state_never_published {
+ color: #FFC100; }
+
+.kitList_state_not_configured {
+ color: #F43D4D; }
+
+.kitList_dropdownButton {
+ display: block;
+ margin: 0 auto; }
+ .kitList_dropdownButton img {
+ position: relative; }
+
+.kitList_noKits {
+ line-height: 5; }
+
+.dropdown-item-button {
+ border-radius: 5px;
+ text-align: center;
+ width: 100%; }
+
+.dropdown-item-button:hover {
+ background-color: #FFC100 !important; }
+
+md-card.kit-list-item {
+ border-radius: 10px; }
+
+/**
+ * USER PROFILE STATE
+ * most styles are re-used on my profile state
+ */
+/* profile header */
+.profile_header {
+ background-color: #2E2E2E;
+ color: #D2D3D5;
+ height: 192px;
+ margin-top: 64px; }
+
+.profile_header_avatar {
+ width: 100px;
+ height: 100px;
+ border-radius: 50px;
+ margin-right: 29px;
+ margin-left: 27px; }
+
+.profile_header_content > div {
+ margin: 5px 0; }
+
+.profile_header_content svg {
+ fill: #ECECEE; }
+
+.profile_header_content_avatar {
+ width: 12px;
+ height: 12px;
+ margin-right: 6px;
+ position: relative;
+ top: -1px; }
+
+.profile_content {
+ min-height: 700px;
+ position: relative; }
+
+/* profile sidebar */
+.profile_sidebar {
+ background-color: #F5F5F5;
+ min-width: 200px; }
+
+.profile_sidebar_title {
+ color: #1C1C1C;
+ font-weight: bold; }
+
+.profile_sidebar_description {
+ font-size: 14px; }
+
+.profile_sidebar_button {
+ height: 64px;
+ color: #0019FF; }
+ .profile_sidebar_button svg {
+ fill: #0019FF; }
+
+.profile_sidebar_avatar {
+ width: 12px;
+ height: 12px;
+ margin-right: 8px;
+ position: relative;
+ top: -2px; }
+
+/* profile main content */
+.profile_content_main_top {
+ margin: 0 0 42px 20px; }
+
+/**
+ * MY PROFILE STATE
+ */
+.myProfile_state md-ink-bar {
+ background-color: #0019FF; }
+
+.myProfile_state md-tab md-tab-label {
+ overflow: visible; }
+
+.myProfile_state md-tabs md-tab {
+ color: #D2D3D5; }
+
+.myProfile_state md-tabs md-tab.active {
+ color: white; }
+
+.myProfile_state md-tabs md-tabs-ink-bar {
+ color: #0019FF;
+ background-color: #0019FF; }
+
+md-tabs-canvas {
+ background-color: #2E2E2E; }
+
+.myProfile_header {
+ height: 188px;
+ position: relative; }
+
+.myProfile_header_container {
+ padding-top: 51px;
+ margin-left: 46px; }
+
+.myProfile_header_avatar {
+ margin-left: 0; }
+
+.myProfile_tabs_parent md-tabs-wrapper {
+ background-color: #1C1C1C; }
+
+.myProfile_tab_icon {
+ width: 16px;
+ height: 16px;
+ margin-right: 8px;
+ position: relative;
+ top: -1px; }
+ .myProfile_tab_icon svg {
+ fill: white; }
+
+.myProfile_sidebar_button {
+ padding-left: 28px; }
+
+.myProfile_content_form {
+ margin-left: 113px;
+ max-width: 500px; }
+
+.myProfile_content_form_input {
+ margin: 16px 0;
+ position: relative; }
+
+.countryInput_container md-autocomplete {
+ background-color: transparent; }
+
+.countryInput_container md-autocomplete-wrap {
+ box-shadow: none !important; }
+
+.countryInput_container input {
+ font-size: 100%; }
+
+.countryInput_container label {
+ position: relative;
+ bottom: 55px; }
+
+.myProfile_form_avatar {
+ margin-bottom: 30px; }
+
+.myProfile_form_avatarImage {
+ width: 64px;
+ height: 64px;
+ border-radius: 32px;
+ margin-right: 7%; }
+
+.myProfile_apiKey_text {
+ color: #D2D3D5;
+ margin-right: 5px;
+ width: 120px; }
+
+.myProfile_apiKey_number {
+ color: #9D9D9E;
+ font-weight: bold;
+ background-color: #F2F2F2;
+ padding: 5px 10px; }
+
+.myProfile_apiKey_refreshButton {
+ background-color: #D2D3D5;
+ height: 29px; }
+ .myProfile_apiKey_refreshButton md-icon {
+ height: 15px;
+ width: 15px;
+ display: block;
+ margin: 3px auto; }
+
+.md-button.myProfile_apiKey_refreshButton {
+ border-radius: 0 4px 4px 0; }
+
+.md-button.myProfile_apiKey_refreshButton:active, .md-button.myProfile_apiKey_refreshButton:hover, .md-button.myProfile_apiKey_refreshButton:focus {
+ background-color: #D2D3D5; }
+
+.myProfile_form_removeButton {
+ display: block;
+ color: #FF3D4C;
+ margin-bottom: 5px; }
+
+.myProfile_updateForm_error {
+ position: absolute;
+ top: 50px;
+ left: 2px;
+ color: red;
+ font-size: 12px;
+ line-height: 1; }
+
+.myProfile_apiKey_block {
+ margin-top: 63px;
+ margin-bottom: 65px; }
+
+.myProfile_apiKey {
+ margin-top: 10px;
+ margin-bottom: 10px; }
+ .myProfile_apiKey svg {
+ fill: #6E6E6E; }
+
+.md-button.myProfile_addKitButton:focus {
+ background-color: #0019FF; }
+
+.md-button.myProfile_addKitButton:hover {
+ background-color: #0019FF; }
+
+.myProfile_hiddenhref {
+ position: absolute;
+ height: 100%;
+ width: 100%;
+ top: 0;
+ left: 0; }
+
+/**
+ * TOOL LIST
+ *
+ */
+a.kitList h4 {
+ color: #1C1C1C; }
+
+a.kitList p {
+ color: #6E6E6E; }
+
+.clearfix {
+ clear: both; }
+
+/*
+ FOOTER COMPONENT
+*/
+footer {
+ z-index: 1;
+ width: 100%;
+ background-color: #1C1C1C; }
+
+.footer_icon {
+ display: block;
+ width: 32px;
+ height: 32px; }
+
+.flag {
+ padding-right: 15px; }
+
+.border-white.footer-block {
+ border: 2px white solid; }
+
+.uptimerobot-logo {
+ width: 150px;
+ justify-self: right;
+ vertical-align: center;
+ padding-top: 10px; }
+
+.uptimerobot-sponsor {
+ display: flex;
+ flex-direction: column;
+ justify-content: start;
+ align-items: flex-start; }
+
+.uptimerobot-sponsor-text p {
+ font-size: 15px !important;
+ margin: 0px; }
+
+@media only screen and (max-width: 950px) {
+ .color-white {
+ text-align: center; }
+ .flag {
+ padding-right: 0px; }
+ .footer-block {
+ margin-bottom: 20px !important; }
+ .sponsor {
+ justify-content: center;
+ align-items: center; } }
+
+@media (max-width: 599px) {
+ .layout-align-xs-start-stretch, .layout-align-xs-center-stretch, .layout-align-xs-end-stretch, .layout-align-xs-space-between-stretch, .layout-align-xs-space-around-stretch {
+ align-items: center; } }
+
+/*
+ RECOVERY PASSWORD STATE
+*/
+.recovery_header {
+ width: 100%;
+ height: 128px;
+ background-color: #1C1C1C; }
+
+.recovery_headerIcon {
+ display: block;
+ width: 50px;
+ height: 50px; }
+
+/*
+ TIMELINE COMPONENT
+
+ used on add and setup kit
+*/
+.timeline {
+ height: 120px;
+ background-color: #2E2E2E;
+ position: fixed;
+ width: 100%;
+ z-index: 10;
+ top: 64px;
+ box-shadow: 0 4px 5px -2px #2E2E2E;
+ padding: 30px; }
+
+@media (max-width: 960px) {
+ .timeline-btn-extra {
+ display: None; } }
+
+@media (min-width: 0px) and (max-width: 959px) and (orientation: portrait) {
+ .timeline {
+ top: 56px; }
+ .profile_header {
+ margin-top: 53px; } }
+
+@media (min-width: 0px) and (max-width: 959px) and (orientation: landscape) {
+ .timeline {
+ top: 53px; }
+ .profile_header {
+ margin-top: 53px; } }
+
+.timeline_container {
+ width: 100%; }
+
+.timeline_line {
+ width: 500px;
+ border: 1px solid #FFC100;
+ position: relative;
+ top: 14px; }
+
+.timeline_line_small {
+ width: 100px !important;
+ margin: 0px 20px;
+ top: 0px !important; }
+
+.timeline-title {
+ text-transform: uppercase;
+ color: #FFC100;
+ font-size: 24px;
+ font-weight: 600; }
+ .timeline-title.vertical {
+ margin-top: 10px; }
+
+.timeline_stepName {
+ text-transform: uppercase;
+ color: #FFC100;
+ font-size: 18px; }
+ .timeline_stepName.vertical {
+ margin-top: 10px; }
+
+.timeline_content {
+ min-height: 400px;
+ width: 100%;
+ margin-top: 220px; }
+
+.md-button.timeline_buttonBack {
+ margin-left: auto;
+ margin-right: 20px; }
+
+section.relaxed-layout {
+ padding: 10px 25px; }
+
+/*
+ ADD KIT STATE
+*/
+.kit_dataChange .form_block {
+ padding: 40px 30px;
+ background-color: #FFF; }
+ .kit_dataChange .form_block.isEven {
+ background-color: #F9F9FB; }
+
+.kit_dataChange .form_blockInput {
+ height: inherit; }
+
+.kit_dataChange .form_blockInput_container {
+ height: inherit; }
+
+.kit_dataChange .form_blockInput_button {
+ height: inherit; }
+
+.kit_dataChange .form_blockInput_map {
+ height: 250px;
+ min-width: 250px; }
+
+.kit_dataChange .form_blockInput_select label {
+ color: #0019FF;
+ margin-right: 10px; }
+
+.kit_dataChange .form_blockInput_chips {
+ margin-top: 35px; }
+
+.kit_dataChange .form_blockContent {
+ padding-top: 5px;
+ float: left; }
+ .kit_dataChange .form_blockContent img, .kit_dataChange .form_blockContent div {
+ float: left; }
+ .kit_dataChange .form_blockContent img {
+ border-radius: 50%;
+ width: 120px;
+ height: 120px;
+ margin: 5px 20px; }
+
+.kit_dataChange .form_blockContent_image {
+ position: relative;
+ top: 3px; }
+
+.kit_dataChange .form_blockContent_text {
+ margin-left: 10px;
+ width: 240px; }
+ .kit_dataChange .form_blockContent_text.long {
+ width: auto !important; }
+ .kit_dataChange .form_blockContent_text h2 {
+ margin-top: 0;
+ margin-bottom: 8px; }
+ .kit_dataChange .form_blockContent_text p {
+ font-size: 14px;
+ color: #A5A5A5;
+ line-height: 1.5; }
+
+.kit_dataChange .form_errors {
+ position: absolute;
+ bottom: 15%;
+ left: 0;
+ color: red;
+ font-size: 0.7em;
+ line-height: 1; }
+
+.emoji {
+ font-weight: 400; }
+
+/*
+ STATIC PAGES
+*/
+.static_page .timeline {
+ align-items: flex-start stretch; }
+ .static_page .timeline .content {
+ padding: 0 0px; }
+
+.static_page .content {
+ max-width: 50em;
+ margin: 0 auto;
+ text-align: justify;
+ padding: 40px 10px; }
+
+.static_page .full-width-img {
+ width: 100%;
+ height: auto;
+ max-width: 2000px;
+ position: relative;
+ margin: 0 0 -5px 0; }
+
+.static_page .embed-container {
+ position: relative;
+ padding-bottom: 56.25%;
+ height: 0;
+ overflow: hidden;
+ max-width: 100%; }
+ .static_page .embed-container iframe, .static_page .embed-container object, .static_page .embed-container embed {
+ position: absolute;
+ top: 0;
+ left: 0;
+ width: 100%;
+ height: 100%; }
+
+.static_page .center {
+ text-align: center;
+ margin-top: 30px; }
+
+.static_page .row {
+ min-height: 200px; }
+ .static_page .row:nth-child(odd) {
+ background-color: #F9F9FB; }
+
+.static_page h1 {
+ color: white; }
+
+.static_page h4 {
+ font-weight: bold;
+ margin-bottom: 0.2em; }
+
+.static_page .md-button.md-primary.md-raised, .static_page .md-button.md-primary.md-fab {
+ padding: 6px 20px;
+ height: 44px;
+ border-radius: 22px;
+ font-size: 18px;
+ background-color: #0019FF;
+ color: white; }
+ .static_page .md-button.md-primary.md-raised:hover, .static_page .md-button.md-primary.md-raised:focus, .static_page .md-button.md-primary.md-fab:hover, .static_page .md-button.md-primary.md-fab:focus {
+ background-color: #3C98AF; }
+ .static_page .md-button.md-primary.md-raised:not([disabled]), .static_page .md-button.md-primary.md-fab:not([disabled]) {
+ box-shadow: none; }
+
+.static_page .md-button.md-primary {
+ color: #0019FF; }
+
+.not-found-404 {
+ height: 36vh; }
+
+@media (min-width: 960px) {
+ .not-found-404 {
+ height: 67.3vh; } }
+
+.sensor-image-margin {
+ margin-top: 40px; }
+
+h3.no-margin {
+ margin: 0; }
+
+p.no-margin {
+ margin: 0; }
+
+.team-cells-margin {
+ margin-bottom: 25px; }
+
+.subtitle-separation {
+ margin-top: 60px; }
+
+.doorbell-button {
+ background-color: #FFC100 !important; }
+
+.styleguide .info-text {
+ color: rgba(0, 0, 0, 0.3); }
+
+.styleguide .info-text-dark {
+ color: rgba(255, 255, 255, 0.3) !important; }
+
+.styleguide .dark-theme-divider {
+ border-top-color: rgba(255, 255, 255, 0.12); }
+
+.styleguide small {
+ margin-top: 20px; }
+
+.styleguide .dark-text-section {
+ background-color: #1C1C1C; }
+ .styleguide .dark-text-section h1 {
+ font-size: 1.8em;
+ color: #fff; }
+ .styleguide .dark-text-section h2 {
+ color: #fff; }
+ .styleguide .dark-text-section h4 {
+ color: #D2D3D5; }
+ .styleguide .dark-text-section h6 {
+ color: #87CCDD; }
+ .styleguide .dark-text-section p {
+ color: #C8E6ED;
+ font-weight: 300; }
+ .styleguide .dark-text-section small {
+ color: #6E6E6E; }
+
+.styleguide .section-padding {
+ padding: 60px; }
+
+.styleguide .colors-section p {
+ padding-left: 20px; }
+
+.styleguide .colors-section .secondary-color {
+ background-color: #6E6E6E; }
+ .styleguide .colors-section .secondary-color p {
+ color: #fff; }
+
+.styleguide .colors-section .terciary_color {
+ background-color: #0019FF; }
+ .styleguide .colors-section .terciary_color p {
+ color: #fff; }
+
+.styleguide .colors-section .secondary_color_light {
+ background-color: #8DB2BA; }
+ .styleguide .colors-section .secondary_color_light p {
+ color: #fff; }
+
+.styleguide .colors-section .secondary-color-pastel {
+ background-color: #C8E6ED; }
+ .styleguide .colors-section .secondary-color-pastel p {
+ color: #8DB2BA; }
+
+.styleguide .colors-section .white {
+ background-color: #fff;
+ border: 1px solid #C8E6ED; }
+ .styleguide .colors-section .white p {
+ color: #8DB2BA; }
+
+.styleguide .colors-section .sensors > div {
+ width: 60px;
+ height: 60px;
+ border-radius: 30px;
+ float: left;
+ margin: 10px; }
+
+.styleguide .colors-section .sensors div:nth-child(1) {
+ background: #ffc107; }
+
+.styleguide .colors-section .sensors div:nth-child(2) {
+ background: #4fc3f7; }
+
+.styleguide .colors-section .sensors div:nth-child(3) {
+ background: #ffee58; }
+
+.styleguide .colors-section .sensors div:nth-child(4) {
+ background: #f06292; }
+
+.styleguide .colors-section .sensors div:nth-child(5) {
+ background: #4caf50; }
+
+.styleguide .colors-section .sensors div:nth-child(6) {
+ background: #8bc34a; }
+
+.styleguide .colors-section .sensors div:nth-child(7) {
+ background: #9575cd; }
+
+.styleguide .colors-section .sensors div:nth-child(8) {
+ background: #fff9c4; }
+
+.styleguide .colors-section .sensors div:nth-child(9) {
+ background: #ffee58; }
+
+.styleguide .colors-section .sensors div:nth-child(10) {
+ background: #ff5722; }
+
+/* Custom Animations */
+.loading-dots span:after {
+ content: '.';
+ -webkit-animation: dots 2s steps(5, end) infinite;
+ animation: dots 2s steps(5, end) infinite; }
+
+@-webkit-keyframes dots {
+ 0%, 20% {
+ color: rgba(0, 0, 0, 0);
+ text-shadow: 0.25em 0 0 rgba(0, 0, 0, 0), 0.5em 0 0 rgba(0, 0, 0, 0); }
+ 40% {
+ color: white;
+ text-shadow: 0.25em 0 0 rgba(0, 0, 0, 0), 0.5em 0 0 rgba(0, 0, 0, 0); }
+ 60% {
+ text-shadow: 0.25em 0 0 #FFC100, 0.5em 0 0 rgba(0, 0, 0, 0); }
+ 80%, 100% {
+ text-shadow: 0.25em 0 0 #FFC100, 0.5em 0 0 #0019FF; } }
+
+@keyframes dots {
+ 0%, 20% {
+ color: rgba(0, 0, 0, 0);
+ text-shadow: 0.25em 0 0 rgba(0, 0, 0, 0), 0.5em 0 0 rgba(0, 0, 0, 0); }
+ 40% {
+ color: white;
+ text-shadow: 0.25em 0 0 rgba(0, 0, 0, 0), 0.5em 0 0 rgba(0, 0, 0, 0); }
+ 60% {
+ text-shadow: 0.25em 0 0 #FFC100, 0.5em 0 0 rgba(0, 0, 0, 0); }
+ 80%, 100% {
+ text-shadow: 0.25em 0 0 #FFC100, 0.5em 0 0 #0019FF; } }
+
+/* Do not remove this comments bellow. It's the markers used by gulp-inject to inject
+ all your sass files automatically */
+
+/*# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImluZGV4LnNjc3MiLCJoZWxwZXJzLnNjc3MiLCJfdmFyaWFibGVzLnNjc3MiLCJjb21wb25lbnRzL2xhbmRpbmcvbGFuZGluZy5zY3NzIiwiY29tcG9uZW50cy9sYXlvdXQvbGF5b3V0LnNjc3MiLCJjb21wb25lbnRzL3NlYXJjaC9zZWFyY2guc2NzcyIsImNvbXBvbmVudHMva2l0L2tpdC5zY3NzIiwiY29tcG9uZW50cy9raXQvc2hvd0tpdC9zaG93S2l0LnNjc3MiLCJjb21wb25lbnRzL3N0b3JlL3N0b3JlTW9kYWwuc2NzcyIsImNvbXBvbmVudHMvdXBsb2FkL2NzdlVwbG9hZC5zY3NzIiwiY29yZS9hbmltYXRpb24vYmFja2Ryb3AvbG9hZGluZ0JhY2tkcm9wLnNjc3MiLCJjb3JlL2FuaW1hdGlvbi9hbmltYXRpb24uZGlyZWN0aXZlLnNjc3MiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsbUpBQVk7QUFDWixpRkFBWTtBQ0FaO0VBQ0UsMEJDU2lCLEVEUmxCOztBQUNEO0VBQ0Usd0JBQXVCLEVBQ3hCOztBQUNEO0VBQ0Usd0JBQXVCLEVBQ3hCOztBQUNEO0VBQ0UsMEJBQStCLEVBQ2hDOztBQUNEO0VBQ0UsZUNIaUIsRURJbEI7O0FBQ0Q7RUFDRSxlQ2RZLEVEZWI7O0FBQ0Q7RUFDRSxlQ2pCWSxFRGtCYjs7QUFDRDtFQUNFLGVDaEJhLEVEb0JkO0VBTEQ7SUFHSSxjQ2xCVyxFRG1CWjs7QUFFSDtFQUNFLGVDdkJXLEVEMkJaO0VBTEQ7SUFHSSx5QkFBcUIsRUFDdEI7O0FBRUg7RUFDRSxZQUFXLEVBQ1o7O0FBQ0Q7RUFDRSxhQUFZLEVBQ2I7O0FBQ0Q7RUFDRSxjQUFZLEVBQ2I7O0FBQ0Q7RUFDRSxZQUFXLEVBQ1o7O0FBQ0Q7RUFDRSxpQkFBZ0IsRUFDakI7O0FBQ0Q7RUFDRSxnQkFBZSxFQUNoQjs7QUFHRDtFQUNFLGdDQUE4QixFQUMvQjs7QUFDRDtFQUNFLHFCQUFtQixFQUNwQjs7QUFFRDtFQUNFLG1CQUFrQixFQUNuQjs7QUFDRDtF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bEdMO0VBcUdLLG9CQUFtQixFQUNuQjs7QUF0R0w7RUF5R0ssb0JBQW1CLEVBQ25COztBQTFHTDtFQTZHSyxvQkFBbUIsRUFDbkI7O0FBOUdMO0VBaUhLLG9CQUFtQixFQUNuQjs7QUFsSEw7RUFxSEssb0JBQW1CLEVBQ25COztBQXRITDtFQXlISyxvQkFBbUIsRUFDbkI7O0FBMUhMO0VBNkhLLG9CQUFtQixFQUNuQjs7QUE5SEw7RUFpSUssb0JBQW1CLEVBQ25COztBQWxJTDtFQXFJSyxvQkFBbUIsRUFDbkI7O0FBdElMO0VBeUlLLG9CQUFtQixFQUNuQjs7QUFNTCx1QkFBdUI7QUFFdkI7RUFDRSxhQUFZO0VBQ1osa0RBQXlDO1VBQXpDLDBDQUF5QyxFQUFFOztBQUU3QztFQUNFO0lBQ0Usd0JBQW9CO0lBQ3BCLHFFQUV3QixFQUFBO0VBQzFCO0lBQ0UsYUFBWTtJQUNaLHFFQUV3QixFQUFBO0VBQzFCO0lBQ0UsNERBRXdCLEVBQUE7RUFDMUI7SUFDRSxtREVqcUVVLEVBQUEsRUFBQTs7QUZpcEVkO0VBQ0U7SUFDRSx3QkFBb0I7SUFDcEIscUVBRXdCLEVBQUE7RUFDMUI7SUFDRSxhQUFZO0lBQ1oscUVBRXdCLEVBQUE7RUFDMUI7SUFDRSw0REFFd0IsRUFBQTtFQUMxQjtJQUNFLG1ERWpxRVUsRUFBQSxFQUFBOztBRnNxRWQ7dUNBQ3VDIiwiZmlsZSI6ImluZGV4LmNzcyIsInNvdXJjZXNDb250ZW50IjpbIkBpbXBvcnQgdXJsKGh0dHBzOi8vZm9udHMuZ29vZ2xlYXBpcy5jb20vY3NzP2ZhbWlseT1Sb2JvdG8rQ29uZGVuc2VkOjQwMCwzMDAsMzAwaXRhbGljLDQwMGl0YWxpYyw3MDAsNzAwaXRhbGljfFJvYm90bzo0MDAsNzAwLDcwMGl0YWxpYyw0MDBpdGFsaWMpO1xuQGltcG9ydCB1cmwoJ2h0dHBzOi8vZm9udHMuZ29vZ2xlYXBpcy5jb20vY3NzP2ZhbWlseT1LYW5pdDo0MDAsNTAwLDYwMCw3MDAsOTAwJyk7XG5cbkBpbXBvcnQgJ192YXJpYWJsZXMnO1xuQGltcG9ydCAnaGVscGVycyc7XG5AaW1wb3J0ICdjb21wb25lbnRzL2xhbmRpbmcvbGFuZGluZyc7XG5AaW1wb3J0ICdjb21wb25lbnRzL2xheW91dC9sYXlvdXQnO1xuQGltcG9ydCAnY29tcG9uZW50cy9zZWFyY2gvc2VhcmNoJztcbkBpbXBvcnQgJ2NvbXBvbmVudHMva2l0L2tpdCc7XG5AaW1wb3J0ICdjb21wb25lbnRzL2tpdC9zaG93S2l0L3Nob3dLaXQnO1xuQGltcG9ydCAnY29tcG9uZW50cy9zdG9yZS9zdG9yZU1vZGFsJztcbkBpbXBvcnQgJ2NvbXBvbmVudHMvdXBsb2FkL2NzdlVwbG9hZCc7XG5AaW1wb3J0ICdjb3JlL2FuaW1hdGlvbi9iYWNrZHJvcC9sb2FkaW5nQmFja2Ryb3AnO1xuQGltcG9ydCAnY29yZS9hbmltYXRpb24vYW5pbWF0aW9uLmRpcmVjdGl2ZSc7XG5cbi8qKlxuICogIE1PWklMTEEtU1BFQ0lGSUMgQ1NTXG4gKi9cbkAtbW96LWRvY3VtZW50IHVybC1wcmVmaXgoKSB7XG4gIC5raXRfZGV0YWlsZWQgLmtpdF9jaGFydCAuY29udGFpbmVyIC5raXRfY2hhcnRfbGVmdCAuc2Vuc29yX3NlbGVjdCAuc2Vuc29yX2ljb25fc2VsZWN0ZWQge1xuICAgIHRyYW5zZm9ybTogdHJhbnNsYXRlWSgtOHB4KTtcbiAgfVxuICAua2l0X2RldGFpbGVkIC5raXRfY2hhcnQgLmNvbnRhaW5lciAua2l0X2NoYXJ0X2xlZnQgLnNlbnNvcl9jb21wYXJlIG1kLXNlbGVjdCB7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKDhweCk7XG4gICAgbWFyZ2luOiAwO1xuICB9XG4gIGJvZHkgLmtpdF9kZXRhaWxlZCAua2l0X2NoYXJ0IC5jb250YWluZXIgLmtpdF9jaGFydF9sZWZ0IC5zZW5zb3JfZGF0YSB7XG4gICAgbWFyZ2luOiA1JSAwIDQlIDA7XG4gIH1cblxuLyogICBzZWN0aW9uLm1hcCB7XG4gICAgei1pbmRleDogMDtcbiAgfSAqL1xufVxuXG4vKipcbiAqICBJTlBVVCBDU1NcbiAqL1xuLy9cbi8vIDo6LXdlYmtpdC1pbnB1dC1wbGFjZWhvbGRlciB7IC8qIFdlYktpdCBicm93c2VycyAqL1xuLy8gICBjb2xvcjogJGZvbnRfY29sb3Jfd2hpdGU7XG4vLyB9XG4vLyA6LW1vei1wbGFjZWhvbGRlciB7IC8qIE1vemlsbGEgRmlyZWZveCA0IHRvIDE4ICovXG4vLyAgIGNvbG9yOiAkZm9udF9jb2xvcl93aGl0ZTtcbi8vICAgb3BhY2l0eTogIDE7XG4vLyB9XG4vLyA6Oi1tb3otcGxhY2Vob2xkZXIgeyAvKiBNb3ppbGxhIEZpcmVmb3ggMTkrICovXG4vLyAgIGNvbG9yOiAgJGZvbnRfY29sb3Jfd2hpdGU7XG4vLyAgIG9wYWNpdHk6ICAxO1xuLy8gfVxuLy8gOi1tcy1pbnB1dC1wbGFjZWhvbGRlciB7IC8qIEludGVybmV0IEV4cGxvcmVyIDEwKyAqL1xuLy8gICBjb2xvcjogJGZvbnRfY29sb3Jfd2hpdGU7XG4vLyB9XG4vL1xuLy8gOmZvY3VzOjotd2Via2l0LWlucHV0LXBsYWNlaG9sZGVyIHsgLyogV2ViS2l0IGJyb3dzZXJzICovXG4vLyAgIGNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2RhcmtfYmFja2dyb3VuZDtcbi8vIH1cbi8vIDpmb2N1czotbW96LXBsYWNlaG9sZGVyIHsgLyogTW96aWxsYSBGaXJlZm94IDQgdG8gMTggKi9cbi8vICAgY29sb3I6ICRiYWNrZ3JvdW5kX2JsdWVfZGFya19iYWNrZ3JvdW5kO1xuLy8gICBvcGFjaXR5OiAgMTtcbi8vIH1cbi8vIDpmb2N1czo6LW1vei1wbGFjZWhvbGRlciB7IC8qIE1vemlsbGEgRmlyZWZveCAxOSsgKi9cbi8vICAgY29sb3I6ICRiYWNrZ3JvdW5kX2JsdWVfZGFya19iYWNrZ3JvdW5kO1xuLy8gICBvcGFjaXR5OiAgMTtcbi8vIH1cbi8vIDpmb2N1czotbXMtaW5wdXQtcGxhY2Vob2xkZXIgeyAvKiBJbnRlcm5ldCBFeHBsb3JlciAxMCsgKi9cbi8vICAgY29sb3I6ICRiYWNrZ3JvdW5kX2JsdWVfZGFya19iYWNrZ3JvdW5kO1xuLy8gfVxuXG5odG1sIHtcbiAgZm9udC1mYW1pbHk6ICdSb2JvdG8gQ29uZGVuc2VkJztcbiAgZm9udC1zaXplOiAxOHB4O1xufVxuXG5ib2R5IHtcbiAgYmFja2dyb3VuZC1jb2xvcjogcmdiYSgwLCAwLCAwLCAwKTtcbiAgY29sb3I6ICRvZmZfYmxhY2s7XG59XG5cbmEuYWJvdXR7XG4gIGN1cnNvcjogcG9pbnRlcjtcbiAgdGV4dC1kZWNvcmF0aW9uOiBub25lO1xuICBjb2xvcjogJGJsdWU7XG59XG5cbmxpLnBvbGljeS10b2Mge1xuICBwYWRkaW5nOiAzcHg7XG4gIGEge1xuICAgIHRleHQtZGVjb3JhdGlvbjogbm9uZTtcbiAgfVxufVxuXG51bCNwb2xpY3ktdG9jIHtcbiAgcGFkZGluZy1ib3R0b206IDBweCAhaW1wb3J0YW50O1xuICBwYWRkaW5nLWJvdHRvbTogMHB4ICFpbXBvcnRhbnQ7XG59XG5cbi5wb2xpY3ktdG9jI2hlYWRlcntcbiAgbGlzdC1zdHlsZTogbm9uZTtcbn1cblxuLmljb25fbGFiZWx7XG4gIHdpZHRoOiAxNHB4O1xuICBoZWlnaHQ6IDE0cHg7XG4gIG1hcmdpbi1yaWdodDogNHB4O1xuICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gIHRvcDogLTJweDtcbiAgc3ZnIHtcbiAgICBmaWxsOiAkZ3JleV9kYXJrZXI7XG4gIH1cbn1cblxuLyoqXG4gICogVFlQT0dSQVBIWVxuICAqL1xuXG5oMXtcbiAgZm9udC1zaXplOiAyLjJlbTtcbiAgZm9udC13ZWlnaHQ6IG5vcm1hbDtcbn1cbmgyIHtcbiAgZm9udC1zaXplOiAxLjdlbTtcbiAgZm9udC13ZWlnaHQ6IG5vcm1hbDtcbn1cbmgzIHtcbiAgZm9udC1zaXplOiAxLjRlbTtcbiAgZm9udC13ZWlnaHQ6IG5vcm1hbDtcbn1cbmg0IHtcbiAgZm9udC1zaXplOiAxLjJlbTtcbiAgZm9udC13ZWlnaHQ6IG5vcm1hbDtcbn1cblxuXG5wIHtcbiAgbGluZS1oZWlnaHQ6IDEuNTtcbn1cblxudGQge1xuICBwYWRkaW5nOjVweDtcbn1cblxuaW5wdXR7XG4gIGZvbnQtZmFtaWx5OidSb2JvdG8gQ29uZGVuc2VkJztcbn1cblxuLmxhYmVsLCAudGFnLCAuc3RhdGV7XG4gIHRleHQtdHJhbnNmb3JtOiB1cHBlcmNhc2U7XG4gIGZvbnQtc2l6ZTogMTJweDtcbiAgbGluZS1oZWlnaHQ6MjRweDtcbiAgYm9yZGVyLXJhZGl1czoyMHB4O1xuICBwYWRkaW5nOiAzcHggOHB4O1xuICBtYXJnaW4tcmlnaHQ6IDhweDtcblxuICBtZC1pY29ue1xuICAgIG1hcmdpbi1sZWZ0OjRweDtcbiAgICBoZWlnaHQ6MTFweDtcbiAgICB3aWR0aDoxMXB4O1xuICAgIG1hcmdpbi10b3A6LTNweDtcbiAgICBjdXJzb3I6cG9pbnRlcjtcbiAgfVxuICAmW2NsaWNrYWJsZV17XG4gICAgY3Vyc29yOnBvaW50ZXI7XG4gIH1cbn1cbi5sYWJlbHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJHllbGxvd19saWdodDtcbiAgY29sb3I6ICRvZmZfYmxhY2s7XG4gIG1kLWljb24gc3ZnIHBhdGh7XG4gICAgZmlsbDogJHRlcmNpYXJ5X2NvbG9yX2xpZ2h0O1xuICB9XG59XG4udGFnIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJHllbGxvd19saWdodGVyO1xuICBjb2xvcjogJG9mZl9ibGFjaztcbiAgbWQtaWNvbiBzdmcgcGF0aHtcbiAgICBmaWxsOiAkc2Vjb25kYXJ5X2NvbG9yO1xuICB9XG59XG5cbi5sZWFmbGV0LWNvbnRhaW5lciAudGFnLCAubGVhZmxldC1jb250YWluZXIgLmxhYmVsLFxuLmxlYWZsZXQtY29udGFpbmVyIC5zdGF0ZXtcbiAgZm9udC1zaXplOjFlbTtcbn1cblxuLmxlYWZsZXQtdG9we1xuICAgdG9wOjkzcHg7XG59XG5cbnNlY3Rpb24uaW5mb3tcbiAgaDF7XG4gICAgZm9udC1zaXplOjEuM2VtO1xuICB9XG59XG5cbmF7XG4gIGNvbG9yOiAkb2ZmX2JsYWNrO1xuICAmOmhvdmVyLCAmOmFjdGl2ZXtcbiAgICB0ZXh0LWRlY29yYXRpb246IG5vbmU7XG4gICAgY29sb3I6ICRibHVlO1xuICB9XG59XG5cbmEuZm9vdGVyLWxpbmt7XG4gIGNvbG9yOiAkeWVsbG93O1xuICB0ZXh0LWRlY29yYXRpb246IG5vbmU7XG4gIHRyYW5zaXRpb246IGNvbG9yIDAuNHMgY3ViaWMtYmV6aWVyKDAuMjUsIDAuOCwgMC4yNSwgMSk7XG4gICY6aG92ZXIsICY6YWN0aXZle1xuICAgIHRleHQtZGVjb3JhdGlvbjogbm9uZTtcbiAgICBjb2xvcjokYmx1ZTtcbiAgfVxufVxuLy8gYS5tZC1idXR0b24sIGEsXG4vLyBidXR0b24ubWQtYnV0dG9uLCBidXR0b257XG4vLyAgIC8vIGNvbG9yIGhlcmUgb3ZlcndyaXRlcyBNQVAgYW5kIENPTU1VTklUWSBpbiBuYXZiYXJcbi8vICAgLy8gY29sb3I6ICR0ZXJjaWFyeV9jb2xvcjtcbi8vIH1cblxuLypcbiAgT3ZlcndyaXRlIGdsb2JhbCBtYXRlcmlhbCBkZXNpZ24sIGFsbG93IGljb25zIHRvIGJlIHNtYWxsZXJcbiovXG5cbm1kLXRvb2xiYXJ7XG4gIGJhY2tncm91bmQtY29sb3I6ICRiYWNrZ3JvdW5kX2JsdWVfZGFya19uYXZiYXIgIWltcG9ydGFudDtcbiAgei1pbmRleDogJGxhcmdlX3ByaW9yaXR5O1xuICBjb2xvcjogd2hpdGUgIWltcG9ydGFudDtcbiAgbWQtaWNvbntcbiAgICBmaWxsOiB3aGl0ZSAhaW1wb3J0YW50O1xuICB9XG4gIC8vaGVpZ2h0OiA2NHB4O1xufVxuXG5tZC1pY29ue1xuICAvLyBFZmZlY3RzIEFMTCBpY29ucyBvbiB0aGUgc2l0ZVxuICBtaW4td2lkdGg6IHVuc2V0ICFpbXBvcnRhbnQ7XG59XG5cbm1kLXNlbGVjdHtcbiAgbWQtc2VsZWN0LXZhbHVle1xuICAgIGJvcmRlci1ib3R0b20tY29sb3I6ICNjY2NjY2M3NSAhaW1wb3J0YW50O1xuICB9XG59XG5cbmEubWQtYnV0dG9uIGg0LCBhLm1kLWJ1dHRvbiAuaDR7XG4gIGNvbG9yOiAkdGVyY2lhcnlfY29sb3I7XG4gIG1hcmdpbjogMDtcbn1cbmEud2FybixcbmJ1dHRvbi53YXJue1xuICBjb2xvcjogJGFsZXJ0X3JlZDtcbiAgLy8gc3ZnIHtcbiAgLy8gICAvLyBmaWxsOiAkYWxlcnRfcmVkO1xuICAvLyB9XG59XG5cbi5ib2xke1xuICBmb250LXdlaWdodDogYm9sZDtcbn1cblxuZGl2W2FwaS1rZXlde1xuICBkaXNwbGF5OiBmbGV4O1xuXG4gIC5hcGlfa2V5X251bWJlciB7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogI0YyRjJGMjtcbiAgICB3b3JkLWJyZWFrOiBicmVhay1hbGw7XG4gICAgd2lkdGg6IDQyMHB4O1xuICB9XG4gIC5hcGlfa2V5X3JlZnJlc2hfYnV0dG9uIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjREJEQkRCO1xuICAgIG1hcmdpbjowO1xuICAgIG1kLWljb24ge1xuICAgICAgcGFkZGluZzogMTBweDtcbiAgICAgIGhlaWdodDogMTVweDtcbiAgICAgIHdpZHRoOiAxNXB4O1xuICAgIH1cbiAgfVxuICBtZC1pY29uIHtcbiAgICBvcGFjaXR5OiAuNTtcbiAgfVxufVxuXG4uaW5mb19vdmVybGF5e1xuICBoMntcbiAgICBjb2xvcjp3aGl0ZTtcbiAgfVxuICBwe1xuICAgIG1hcmdpbjogMDtcbiAgICBmb250LXNpemU6IDIwcHg7XG4gICAgZm9udC13ZWlnaHQ6IDMwMDtcbiAgICBjb2xvcjokdGVyY2lhcnlfY29sb3JfbGlnaHQ7XG4gIH1cbn1cblxuLmRhcmt7XG4gIGgxLCBoMntcbiAgICBjb2xvcjp3aGl0ZTtcbiAgfVxuICBoMSwgaDJ7XG4gICAgbWFyZ2luOjA7XG4gIH1cbiAgaDMsIGg0LCAuaDQsIGF7XG4gICAgY29sb3I6ICRmb250X2NvbG9yX2JsdWVfZ3JleTtcbiAgfVxufVxuXG4vKipcbiAqIEFOR1VMQVIgTUFURVJJQUwgT1ZFUlJJRElOR1xuICovXG5cblxubWQtc2VsZWN0LW1lbnUge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiB3aGl0ZTtcbn1cbm1kLXNlbGVjdCxcbm1kLXNlbGVjdC1tZW51IHtcbiAgdGV4dC10cmFuc2Zvcm06IHVwcGVyY2FzZTtcbn1cbm1kLXNlbGVjdDpub3QoW2Rpc2FibGVkXSk6Zm9jdXMgLm1kLXNlbGVjdC12YWx1ZSB7XG4gIGJvcmRlci1ib3R0b20tY29sb3I6ICM2Qjg2OEQ7XG59XG5cbi8vIGJ1dHRvbi5tZC1wcmltYXJ5e1xuLy8gICAvLyBHZXQgbmV3IGtpdCBwb3B1cCwgYnV0dG9uIGNvbG9yXG4vLyAgIC8vIEFsc28gaW50ZXJmZXJlcyB3aXRoIGtpdHMvZWRpdCBzYXZlIGJ1dHRvblxuLy8gICAvLyBjb2xvcjokdGVyY2lhcnlfY29sb3IgIWltcG9ydGFudDtcbi8vICAgLy8gZm9udC1zaXplOjE4cHg7XG4vLyB9XG4vLyAubWQtYnV0dG9uLm1kLWZsYXQubWQtcHJpbWFyeSB7XG4vLyAgIC8vbWFyZ2luOiAwcHg7XG4vLyAgIC8vYmFja2dyb3VuZC1jb2xvcjogJHRlcmNpYXJ5X2NvbG9yO1xuLy8gICAvL2NvbG9yOiB3aGl0ZSAhaW1wb3J0YW50O1xuLy8gfVxuXG4ubWQtYnV0dG9uLm1kLXByaW1hcnkubWQtZmxhdDpub3QoW2Rpc2FibGVkXSk6aG92ZXIge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkdGVyY2lhcnlfY29sb3I7XG4gIGNvbG9yOiB3aGl0ZTtcbn1cblxuLm1kLWJ1dHRvbi5tZC1wcmltYXJ5Lm1kLWZsYXQ6bm90KFtkaXNhYmxlZF0pOmZvY3VzIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJHRlcmNpYXJ5X2NvbG9yO1xuICBjb2xvcjogd2hpdGU7XG59XG5cbi5tZC1idXR0b24uZmlsbHdpZHRoIHtcbiAgd2lkdGg6IDEwMCUgIWltcG9ydGFudDtcbn1cblxubWQtaW5wdXQtY29udGFpbmVyOm5vdCgubWQtaW5wdXQtaW52YWxpZCkubWQtaW5wdXQtZm9jdXNlZCB7XG4gIGxhYmVsIHtcbiAgICBjb2xvcjogJHRlcmNpYXJ5X2NvbG9yO1xuICB9XG5cbiAgLm1kLWlucHV0IHtcbiAgICBib3JkZXItY29sb3I6IHJnYmEoMCwwLDAsMC4xMik7XG4gIH1cbn1cblxuLyoqXG4gKiBBTEVSVCBDT01QT05FTlRcbiAqL1xuXG4ubWQtdG9hc3QtY29udGVudCB7XG4gIGJveC1zaGFkb3c6IHVuc2V0ICFpbXBvcnRhbnQ7XG4gIGJhY2tncm91bmQtY29sb3I6IHVuc2V0ICFpbXBvcnRhbnQ7XG4gIGNvbG9yOiBibGFjayAhaW1wb3J0YW50O1xuICBtaW4td2lkdGg6IDEwMCU7XG59XG5cbm1kLXRvYXN0IHtcbiAgLy8gVE9ETzogQnVnIHdoZW4gYSB0b2FzdCBpcyBvbiBzY3JlZW4sIHRoZSBib2R5IGdldHMgY2xhc3M9J21kLXRvYXN0LW9wZW4tYm90dG9tJ1xuICAvLyBXaGVyZSBkb2VzIHRoYXQgY29tZSBmcm9tP1xuICAvLyBJZiB0aGVyZSBpcyBhIHNpbWlsYXIgKm9wZW4tdG9wLCB3ZSBjYW4gcmVtb3ZlIHRoZSBDU1MgYm90dG9tLCB0b3BcbiAgYm90dG9tOiB1bnNldCAhaW1wb3J0YW50OyAvLyBPdmVyd3JpdGUgYW5ndWxhciBkZWZhdWx0XG4gIGNvbG9yOiBibGFjaztcbiAgb3BhY2l0eTogMC45NTtcbiAgcGFkZGluZzogMDtcbiAgcG9zaXRpb246IGZpeGVkO1xuICB0b3A6IDY0cHggIWltcG9ydGFudDsgLy8gU3RhcnQgdGhlIHRvYXN0IGF0IHRvcCBiZWxvdyBuYXZiYXIuXG4gIG1pbi13aWR0aDogMTAwJTtcbiAgei1pbmRleDogMTU7XG5cbiAgLmFsZXJ0X3R5cGVJY29uIHtcbiAgICB3aWR0aDogMTZweDtcbiAgICBoZWlnaHQ6IDE2cHg7XG4gICAgbWFyZ2luLXJpZ2h0OiAxNnB4O1xuICB9XG5cbiAgLmFsZXJ0X2Nsb3NlSWNvbiB7XG4gICAgd2lkdGg6IDEwLjVweDtcbiAgICBoZWlnaHQ6IDEwLjVweDtcbiAgICBtYXJnaW46IDAgYXV0bztcbiAgfVxuXG4gIC5hbGVydF9idXR0b24ge1xuICAgIHBhZGRpbmc6IDBweCAxMnB4O1xuICAgIG1hcmdpbi1sZWZ0OiA1cHg7XG4gICAgY29sb3I6IGJsYWNrICFpbXBvcnRhbnQ7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogJHllbGxvdztcbiAgICBib3JkZXItcmFkaXVzOiAyMHB4O1xuICAgIGZvbnQtc2l6ZTogMTJweDtcbiAgfVxufVxuXG5tZC10b2FzdC5yZWQge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkYWxlcnRfcmVkO1xufVxuXG5tZC10b2FzdC5ncmVlbiB7XG4gIGJhY2tncm91bmQtY29sb3I6ICRhbGVydF9ncmVlbjtcbn1cblxubWQtdG9hc3QuYmx1ZSB7XG4gIGJhY2tncm91bmQtY29sb3I6ICRibHVlO1xufVxuXG5tZC10b2FzdC55ZWxsb3cge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeWVsbG93O1xufVxuXG5tZC1jaGVja2JveC5tZC1jaGVja2VkIC5tZC1pY29uIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmx1ZTtcbn1cbm1kLWNoZWNrYm94IC5tZC1sYWJlbCB7XG4gICAgcG9pbnRlci1ldmVudHM6IGF1dG87XG59XG5tZC1jaGVja2JveCAubWQtbGFiZWwgc3BhbiB7XG4gICAgcG9pbnRlci1ldmVudHM6IG5vbmU7XG4gICAgLXdlYmtpdC11c2VyLXNlbGVjdDogbm9uZTtcbiAgICAtbW96LXVzZXItc2VsZWN0OiBub25lO1xuICAgIC1tcy11c2VyLXNlbGVjdDogbm9uZTtcbiAgICB1c2VyLXNlbGVjdDogbm9uZTtcbn1cblxuLyoqXG4gKiAgU1BJTk5FUi9QUk9HUkVTUyBCQVIgQ09NUE9ORU5UU1xuICovXG5cbm1kLXByb2dyZXNzLWxpbmVhci5tZC1iYXIuZ3JlZW4ge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkZ3JlZW47XG59XG5tZC1wcm9ncmVzcy1saW5lYXIuZ3JlZW4gLm1kLWNvbnRhaW5lciB7XG4gIGJhY2tncm91bmQtY29sb3I6IHJnYmEoJGdyZWVuLCAwKTtcbn1cbm1kLXByb2dyZXNzLWxpbmVhci5ncmVlbiAubWQtYmFyIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJGdyZWVuO1xufVxubWQtcHJvZ3Jlc3MtbGluZWFyLm1kLWJhcntcbiAgYmFja2dyb3VuZC1jb2xvcjogJGJsdWU7XG59XG5tZC1wcm9ncmVzcy1saW5lYXIubWQtaHVlLTMgLm1kLWNvbnRhaW5lcntcbiAgYmFja2dyb3VuZC1jb2xvcjogcmdiYSgkYmx1ZSwgMC4wNSk7XG59XG5tZC1wcm9ncmVzcy1saW5lYXIubWQtaHVlLTMgLm1kLWJhciB7XG4gIGJhY2tncm91bmQtY29sb3I6ICRibHVlO1xufVxuXG4ua2l0X3NwaW5uZXIge1xuICB6LWluZGV4OiAyMDA7XG4gIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgbGVmdDogMDtcbiAgYm90dG9tOiA2cHg7XG59XG5cbi5jaGFydF9zcGlubmVyIHtcbiAgcG9zaXRpb246IGFic29sdXRlO1xuICB0b3A6IDIwMHB4O1xuICBsZWZ0OiA1NyU7XG59XG5cbi8qKlxuICogIENPT0tJRVMgUE9MSUNZIENPTVBPTkVOVFxuICovXG5cbiAuY29va2llcy1wb2xpY3lfY29udGFpbmVye1xuICAgIGJhY2tncm91bmQtY29sb3I6IHJnYmEoNjAsIDYwLCA2MCwgMSk7XG4gICAgY29sb3I6ICR5ZWxsb3c7XG4gICAgcG9zaXRpb246IGZpeGVkO1xuICAgIGJvdHRvbTogMDtcbiAgICBwYWRkaW5nOjVweDtcbiAgICBsZWZ0OiAwO1xuICAgIHdpZHRoOiAxMDAlO1xuICAgIC8vaGVpZ2h0OiA0dmg7XG4gICAgei1pbmRleDogOTk5O1xuICAgIHRleHQtYWxpZ246IGNlbnRlcjtcbiAgICBsaW5lLWhlaWdodDogNHZoO1xuXG4gICAgYXtcbiAgICAgIGNvbG9yOiByZ2JhKCR5ZWxsb3csIDAuNik7XG4gICAgfVxuIH1cblxuXG4vLyBObyBkYXRhIGJhY2tkcm9wXG4ubWQtbm9EYXRhQmFja2Ryb3Age1xuICB3aWR0aDogMTAwJTtcbiAgaGVpZ2h0OiA5NiArIDI4NXB4O1xuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xuXG4gIC5ibG9jayB7XG4gICAgbWFyZ2luOiAwIGF1dG87XG5cbiAgfVxuICAudGl0bGUsIC5tZXNzYWdlIHtcbiAgICB0ZXh0LWFsaWduOiBjZW50ZXI7XG4gIH1cbiAgLnRpdGxlIHtcbiAgICBjb2xvcjogJGdyZXlfZGFya2VyO1xuICAgIGZvbnQtd2VpZ2h0OiA0MDA7XG4gICAgbWFyZ2luLWJvdHRvbTogMDtcbiAgfVxuICAubWVzc2FnZSB7XG4gICAgbWFyZ2luOiAwO1xuICAgIGZvbnQtc2l6ZTogMjBweDtcbiAgICBmb250LXdlaWdodDogMzAwO1xuICB9XG4gIGEge1xuICAgIGNvbG9yOiAkYmx1ZTtcbiAgICB0ZXh0LWRlY29yYXRpb246IG5vbmU7XG4gIH1cbn1cblxuLyoqXG4gKiAgTUFQXG4gKi9cblxuLm1hcF9zdGF0ZSB7XG4gIHBvc2l0aW9uOiBmaXhlZDtcbiAgei1pbmRleDogMTtcbn1cblxuLy8gQG1lZGlhIG9ubHkgc2NyZWVuIGFuZCAobWF4LWRldmljZS13aWR0aDogMTAyNHB4KSB7XG4vLyAgIC5tYXBfc3RhdGUge1xuLy8gICB9XG4vLyB9XG5cbnNlY3Rpb24ubWFwIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogd2hpdGU7XG4gIHRleHQtYWxpZ246IGNlbnRlcjtcbiAgd2lkdGg6IDEwMCU7XG4gIGhlaWdodDogMTAwJTtcbiAgY29sb3I6IGJsYWNrO1xuICB6LWluZGV4OiAxOy8vJHNtYWxsX3ByaW9yaXR5O1xuICBwb3NpdGlvbjogZml4ZWQ7XG4gIHRvcDogNjRweDtcblxuICBAbWVkaWEgYWxsIGFuZCAobWF4LWhlaWdodDogMTIwMHB4KSBhbmQgKG1pbi1oZWlnaHQ6IDEwMDBweCkge1xuICAgIGhlaWdodDogOTUwcHg7XG4gIH1cbiAgQG1lZGlhIGFsbCBhbmQgKG1heC1oZWlnaHQ6IDEwMDBweCkgYW5kIChtaW4taGVpZ2h0OiA4MDBweCkge1xuICAgIGhlaWdodDogNzAwcHg7XG4gIH1cbiAgQG1lZGlhIGFsbCBhbmQgKG1heC1oZWlnaHQ6IDgwMHB4KSBhbmQgKG1pbi1oZWlnaHQ6IDYwMHB4KSB7XG4gICAgaGVpZ2h0OiA1NTBweDtcbiAgfVxuICBAbWVkaWEgYWxsIGFuZCAobWF4LWhlaWdodDogNjAwcHgpIGFuZCAobWluLWhlaWdodDogNDAwcHgpIHtcbiAgICBoZWlnaHQ6IDMwMHB4O1xuICB9XG5cblxuICBoMSB7XG4gICAgZm9udC1zaXplOiAzZW07XG4gIH1cbn1cblxuLypcbiAgQ09SRSBDSElQIENPTVBPTkVOVFxuICovXG4uY2hpcHMge1xuICBkaXNwbGF5OiBpbmxpbmUtYmxvY2s7XG5cbiAgZGl2LmNoaXBzX3Jvd3tcbiAgICBoZWlnaHQ6NDAuNXB4O1xuICB9XG5cbiAgLmxhYmVsLCAudGFne1xuICAgIGJveC1zaGFkb3c6IDAgMXB4IDVweCByZ2JhKDAsMCwwLDAuNjUpO1xuICB9XG59XG4uY2hpcCB7XG4gIC8vIGRpc3BsYXk6IGlubGluZS1ibG9jaztcbiAgcGFkZGluZzogMHB4IDEwcHggMHB4IDE0cHg7XG4gIG1hcmdpbjogYXV0byA1cHg7XG4gIGJvcmRlci1yYWRpdXM6IDIwcHg7XG4gIGhlaWdodDozMHB4O1xuICB0ZXh0LXRyYW5zZm9ybTogdXBwZXJjYXNlO1xuICBmb250LXNpemU6IDEzcHg7XG4gIGxpbmUtaGVpZ2h0OiAyOXB4O1xuXG4gIC5jaGlwX25hbWUge1xuICAgIG1hcmdpbi1sZWZ0OiAzcHg7XG4gICAgbWFyZ2luLXJpZ2h0OiA3cHg7XG4gIH1cbiAgLmNoaXBfaWNvbiB7XG4gICAgYm9yZGVyOiAwO1xuICAgIGJhY2tncm91bmQtY29sb3I6IGluaGVyaXQ7XG4gICAgbWQtaWNvbiB7XG4gICAgICB3aWR0aDogOHB4O1xuICAgICAgaGVpZ2h0OiA4cHg7XG4gICAgfVxuICB9XG5cbiAgJi5tYXBfZmlsdGVyIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiB3aGl0ZTtcbiAgICBib3gtc2hhZG93OiAwIDFweCA1cHggcmdiYSgwLDAsMCwwLjY1KTtcbiAgfVxuICAmLmtpdF90YWcge1xuICAgIGJhY2tncm91bmQtY29sb3I6ICRncmV5O1xuICAgIGZsb2F0OiBsZWZ0O1xuICAgIG1hcmdpbjogNXB4O1xuICB9XG59XG4vKlxuICBNQVAgRklMVEVSU1xuKi9cbi5tYXBfbGVnZW5kIHtcblxuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHRvcDogMTBweDtcbiAgbWFyZ2luLWxlZnQ6IDEwcHg7XG4gIGRpc3BsYXk6IGZsZXg7XG5cbiAgLmNoaXBze1xuICAgIGFsaWduLXNlbGY6IGZsZXgtZW5kO1xuICAgIHBhZGRpbmctYm90dG9tOiA1cHg7XG4gICAgcGFkZGluZy1sZWZ0OiA1cHg7XG4gICAgZGlzcGxheTogZmxleDtcbiAgfVxuXG4gIC5tYXBfbGVnZW5kX19maWx0ZXJzQ29udGFpbmVye1xuICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgZmxleC1kaXJlY3Rpb246IGNvbHVtbjtcbiAgICBqdXN0aWZ5LWNvbnRlbnQ6IHNwYWNlLWJldHdlZW47XG4gICAgd2lkdGg6IDEyNHB4O1xuICAgIGhlaWdodDogODFweDtcblxuICAgIGJvcmRlci1yYWRpdXM6NHB4O1xuICAgIG92ZXJmbG93OmhpZGRlbjtcbiAgICBib3gtc2hhZG93OiAwIDFweCA1cHggcmdiYSgwLCAwLCAwLCAwLjY1KTtcbiAgICBzdmcsIHAge1xuICAgICAgZmlsbDogJG9mZl9ibGFjaztcbiAgICB9XG5cblxuICAgIC5tYXBfbGVnZW5kX19maWx0ZXJzUm93e1xuICAgICAgY3Vyc29yOiBwb2ludGVyO1xuXG4gICAgICAmOm5vdCg6bGFzdC1jaGlsZCl7XG4gICAgICAgIGJvcmRlci1ib3R0b206IDFweCBzb2xpZCBncmF5O1xuICAgICAgfVxuXG4gICAgICBkaXNwbGF5OiBmbGV4O1xuXG4gICAgICBhbGlnbi1pdGVtczogY2VudGVyO1xuICAgICAgcGFkZGluZy1sZWZ0OiA1cHg7XG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjZmZmO1xuXG4gICAgICAubWFwX2ZpbHRlcl9idXR0b257XG4gICAgICAgIGJveC1zaGFkb3c6bm9uZTtcbiAgICAgICAgcHtcbiAgICAgICAgICBtYXJnaW46IDA7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG4gIH1cbn1cbi5tYXBfZmlsdGVyX2J1dHRvbiB7XG4gIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xuICBtYXJnaW4tcmlnaHQ6IDE1cHg7XG4gIHdpZHRoOiAyNnB4O1xuICBoZWlnaHQ6IDI2cHg7XG4gIGJveC1zaGFkb3c6IDAgMXB4IDVweCByZ2JhKDAsMCwwLDAuNjUpO1xuXG4gIG1kLWljb24ge1xuICAgIHdpZHRoOiAxOHB4O1xuICAgIGhlaWdodDogMTJweDtcbiAgfVxufVxuLm1kLWJ1dHRvbi5tYXBfZmlsdGVyX2J1dHRvbjpob3ZlciwgLm1kLWJ1dHRvbi5tYXBfZmlsdGVyX2J1dHRvbjpmb2N1cyB7XG4gIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xufVxuXG5cblxuLypcbj09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5LSVQgVEFHUyBNT0RVTEVcbj09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4qL1xuXG4ua2l0VGFnc19fc2VjdGlvbntcbiAgbWFyZ2luLXRvcDogMzc2cHg7XG5cbiAgLmtpdFRhZ3NfX2xpc3RDb250YWluZXJ7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogJGdyZXlfbGlnaHRlcjtcbiAgICBwYWRkaW5nOiAyMHB4O1xuICB9XG5cbiAgLmtpdFRhZ3NfX2NvbnRhaW5lcntcbiAgICBwYWRkaW5nLWJvdHRvbTogMjBweDtcblxuICAgIC5raXRUYWdzX190ZXh0Q29udGFpbmVye1xuICAgICAgZGlzcGxheTogZmxleDtcbiAgICAgIGp1c3RpZnktY29udGVudDogc3BhY2UtYmV0d2VlbjtcbiAgICAgIG1hcmdpbi1sZWZ0OiAyMHB4O1xuICAgICAgbWFyZ2luLXJpZ2h0OiAyMHB4O1xuICAgICAgLmtpdFRhZ3NfX3RleHRFbGVtZW50e1xuICAgICAgICBmbGV4LWJhc2lzOjQwJTtcbiAgICAgICAgZmxleC1ncm93OjA7XG4gICAgICAgIG1pbi13aWR0aDogMzAwcHg7XG4gICAgICB9XG4gICAgfVxuICB9XG59XG5cbi8qXG4gQ09SRSBGT1JNIENPTVBPTkVOVFxuICovXG5cbi5mb3JtX2NvbnRhaW5lciB7XG4gIGZvbnQtc2l6ZTogMTZweDtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xuICBib3JkZXItcmFkaXVzOiAwcHggMHB4IDNweCAzcHg7XG59XG4uZm9ybV9jb250ZW50Q29udGFpbmVyIHtcbiAgd2lkdGg6IDkyJTtcbiAgbWFyZ2luOiAwIGF1dG87XG59XG4uZm9ybV9maWVsZCB7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbn1cblxuLmZvcm1fZXJyb3JzIHtcbiAgcG9zaXRpb246IGFic29sdXRlO1xuICBib3R0b206IDE1JTtcbiAgbGVmdDogNSU7XG4gIGNvbG9yOiByZWQ7XG4gIGZvbnQtc2l6ZTogMC43ZW07XG4gIGxpbmUtaGVpZ2h0OiAxO1xufVxuXG4uZm9ybV9jbG9zZUljb24ge1xuICBmbG9hdDogcmlnaHQ7XG4gIHdpZHRoOiAxNHB4O1xuICBoZWlnaHQ6IDE0cHg7XG4gIG1hcmdpbjogMTJweDtcblxuICBtZC1pY29uIHtcbiAgICB3aWR0aDogMTRweDtcbiAgICBoZWlnaHQ6IDE0cHg7XG4gICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgIHRvcDogMHB4O1xuICAgIGxlZnQ6IDBweDtcbiAgfVxufVxuXG4ubWQtcHJpbWFyeS5mb3JtX2J1dHRvbiB7XG4gIGNvbG9yOiAkZm9udF9jb2xvcl93aGl0ZTtcbiAgYmFja2dyb3VuZC1jb2xvcjogJHRlcmNpYXJ5X2NvbG9yO1xuICBoZWlnaHQ6IDY0cHg7XG4gIGJvdHRvbTogMHB4O1xuICBsZWZ0OiAwcHg7XG4gIGJvcmRlci1yYWRpdXM6IDBweCAwcHggMnB4IDJweDtcbn1cblxuXG4uZm9ybV90aXRsZSB7XG4gIGNvbG9yOiAkdGVyY2lhcnlfY29sb3I7XG4gIGZvbnQtd2VpZ2h0OiBub3JtYWw7XG59XG5cbi5mb3JtX21lc3NhZ2VIZWFkZXIge1xuICBmb250LXNpemU6IDEuNWVtO1xuICBjb2xvcjogJHNlY29uZGFyeV9jb2xvcjtcbiAgZm9udC13ZWlnaHQ6IGJvbGQ7XG4gIGxpbmUtaGVpZ2h0OiAwO1xuICBtYXJnaW4tdG9wOiA1MHB4O1xufVxuLmZvcm1fbWVzc2FnZVN1YmhlYWRlciB7XG4gIGZvbnQtc2l6ZTogMC44NzVlbTtcbiAgY29sb3I6IHJnYmEoMCw5OCwxMjMsMC41NCk7XG4gIGxpbmUtaGVpZ2h0OiAwLjU7XG59XG5cbi5mb3JtX21lc3NhZ2VEZXNjcmlwdGlvbiB7XG4gIGZvbnQtc2l6ZTogMTJweDtcbiAgbWFyZ2luLWJvdHRvbTogMDtcbn1cblxuLypcbiAgQ09SRSBMQVJHRSBERVZJQ0VTIEZPUk1cbiovXG5AbWVkaWEgKG1pbi13aWR0aDogNTAxcHgpIHtcbiAgLmZvcm1fY29udGFpbmVyIHtcbiAgICB3aWR0aDogMzQwcHg7XG4gICAgZm9udC1zaXplOiAxNnB4O1xuICB9XG4gIC5mb3JtX21lc3NhZ2VDb250YWluZXIge1xuICAgIG1hcmdpbi1ib3R0b206IDIwcHg7XG4gIH1cbiAgLm1kLXByaW1hcnkuZm9ybV9idXR0b24ge1xuICAgIHdpZHRoOiAzNDBweDtcbiAgICBkaXNwbGF5OiBibG9jaztcbiAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gICAgbWFyZ2luOiAwIGF1dG87XG4gICAgZm9udC1zaXplOiAxLjVlbTtcbiAgfVxufVxuXG4vKlxuICBDT1JFIE1PQklMRSBERVZJQ0VTIEZPUk1cbiovXG5cbkBtZWRpYSAobWF4LXdpZHRoOiA1MDBweCkge1xuICAuZm9ybV9jb250YWluZXIge1xuICAgIHdpZHRoOiA4MCU7XG4gICAgZm9udC1zaXplOiAxMnB4O1xuICB9XG4gIC5tZC1wcmltYXJ5LmZvcm1fYnV0dG9uIHtcbiAgICB3aWR0aDogMTAwJTtcbiAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgZm9udC1zaXplOiAxLjJlbTtcbiAgfVxufVxuXG4vLyBvdmVycmlkZSBmb3IgSXBob25lIDRcbkBtZWRpYSBvbmx5IHNjcmVlbiBhbmQgKG1pbi1kZXZpY2Utd2lkdGg6IDMyMHB4KSBhbmQgKG1heC1kZXZpY2Utd2lkdGg6IDQ4MHB4KSBhbmQgKC13ZWJraXQtZGV2aWNlLXBpeGVsLXJhdGlvOiAyKSBhbmQgKGRldmljZS1hc3BlY3QtcmF0aW86IDIvMykge1xuICAucmVjb3ZlcnlfY29udGFpbmVyIHtcbiAgICBtYXJnaW4tdG9wOiA0MHB4O1xuICB9XG4gIC5mb3JtX2ZpZWxkIHtcbiAgICBwYWRkaW5nOiA1JSAwO1xuICB9XG59XG5cbi8qXG4gIFJFQ09WRVJZIEZPUk1cbiovXG4ucmVjb3ZlcnlfY29udGFpbmVyIHtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xuICBtYXJnaW46IDBweCBhdXRvO1xuICBtYXJnaW4tYm90dG9tOjEwMHB4O1xuLy8gIGhlaWdodDogMzc2cHg7XG59XG5cbi5mb3JtUmVjb3ZlcnlfZmllbGQge1xuICBwYWRkaW5nOiAxMCUgMDtcbn1cblxuQG1lZGlhIChtaW4td2lkdGg6IDUwMXB4KSB7XG4gIC5yZWNvdmVyeV9jb250YWluZXIge1xuICAgIG1hcmdpbi10b3A6IDgwcHg7XG4gIH1cbiAgLmZvcm1SZWNvdmVyeV9lcnJvcnMge1xuICAgIGJvdHRvbTogMTUlO1xuICAgIGxlZnQ6IDUlO1xuICB9XG59XG5cbkBtZWRpYSAobWF4LXdpZHRoOiA1MDBweCkge1xuICAucmVjb3ZlcnlfY29udGFpbmVyIHtcbiAgICBtYXJnaW4tdG9wOiAyMCU7XG4gIH1cbiAgLmZvcm1SZWNvdmVyeV9lcnJvcnMge1xuICAgIGJvdHRvbTogMjMlO1xuICAgIGxlZnQ6IDUlO1xuICB9XG59XG5cbkBtZWRpYSBvbmx5IHNjcmVlbiBhbmQgKG1pbi1kZXZpY2Utd2lkdGg6IDMyMHB4KSBhbmQgKG1heC1kZXZpY2Utd2lkdGg6IDQ4MHB4KSBhbmQgKC13ZWJraXQtZGV2aWNlLXBpeGVsLXJhdGlvOiAyKSBhbmQgKGRldmljZS1hc3BlY3QtcmF0aW86IDIvMykge1xuXG4gIEBtZWRpYSAobWF4LXdpZHRoOiA1MDBweCkge1xuICAgIC5mb3JtUmVjb3ZlcnlfZmllbGQge1xuICAgICAgcGFkZGluZzogMiUgMDtcbiAgICB9XG4gICAgLmZvcm1SZWNvdmVyeV9lcnJvcnMge1xuICAgICAgYm90dG9tOiAyOSU7XG4gICAgfVxuICB9XG59XG5cbi8qXG4gIFJFU0VUIEZPUk1cbiovXG4uZm9ybVJlc2V0X2ZpZWxkIHtcbiAgcGFkZGluZzogNyUgMDtcbn1cblxuQG1lZGlhIChtaW4td2lkdGg6IDUwMXB4KSB7XG4gIC5mb3JtUmVzZXRfZXJyb3JzIHtcbiAgICBib3R0b206IDUlO1xuICAgIGxlZnQ6IDUlO1xuICB9XG59XG5cbkBtZWRpYSAobWF4LXdpZHRoOiA1MDBweCkge1xuICAuZm9ybVJlc2V0X2Vycm9ycyB7XG4gICAgYm90dG9tOiAxOCU7XG4gICAgbGVmdDogMSU7XG4gIH1cbn1cblxuQG1lZGlhIG9ubHkgc2NyZWVuIGFuZCAobWluLWRldmljZS13aWR0aDogMzIwcHgpIGFuZCAobWF4LWRldmljZS13aWR0aDogNDgwcHgpIGFuZCAoLXdlYmtpdC1kZXZpY2UtcGl4ZWwtcmF0aW86IDIpIGFuZCAoZGV2aWNlLWFzcGVjdC1yYXRpbzogMi8zKSB7XG4gIEBtZWRpYSAobWF4LXdpZHRoOiA1MDBweCkge1xuICAgIC5mb3JtUmVzZXRfZXJyb3JzIHtcbiAgICAgIHRvcDogNjUlO1xuICAgIH1cbiAgICAuZm9ybVJlc2V0X2ZpZWxkIHtcbiAgICAgIHBhZGRpbmc6IDMlIDA7XG4gICAgfVxuICB9XG59XG5cbi8qKlxuICogTU9WRSBET1dOIEVGRkVDVCBPTiBTQ1JPTExcbiAqL1xuXG4ubW92ZV9kb3duIHtcbiAgLW1zLXRyYW5zZm9ybTogdHJhbnNsYXRlWSgzMnB4KTtcbiAgLXdlYmtpdC10cmFuc2Zvcm06IHRyYW5zbGF0ZVkoMzJweCk7XG4gIC1tb3otdHJhbnNmb3JtOiB0cmFuc2xhdGVZKDMycHgpO1xuICAtby10cmFuc2Zvcm06IHRyYW5zbGF0ZVkoMzJweCk7XG4gIHRyYW5zZm9ybTogdHJhbnNsYXRlWSgzMnB4KTtcbn1cblxuLypcbiAgTUFSS0VSIEFORCBQT1BVUCBNQVJLRVJcbiovXG5cbi5tYXJrZXJTbWFydENpdGl6ZW5Ob3JtYWwge1xuICBib3JkZXI6IDFweCBzb2xpZCAkeWVsbG93O1xuICBib3JkZXItcmFkaXVzOiAyMHB4O1xuICBib3gtc2hhZG93OiAwcHggMnB4IDRweCAwcHggcmdiYSgwLDAsMCwwLjMwKTtcbn1cblxuLm1hcmtlclNtYXJ0Q2l0aXplbk9ubGluZSB7XG4gIGJvcmRlcjogMnB4IHNvbGlkICR5ZWxsb3c7XG4gIGJvcmRlci1yYWRpdXM6IDIwcHg7XG4gIGJveC1zaGFkb3c6IDBweCAycHggNHB4IDBweCByZ2JhKDAsMCwwLDAuMzApO1xuICBiYWNrZ3JvdW5kLWNvbG9yOiByZ2JhKCR5ZWxsb3csIDAuNSk7XG59XG5cbi5tYXJrZXJFeHBlcmltZW50YWxOb3JtYWwge1xuICBib3JkZXI6IDJweCBzb2xpZCAkYmFja2dyb3VuZF9ibHVlX21pZHRvbmU7XG4gIGJvcmRlci1yYWRpdXM6IDIwcHg7XG4gIGJveC1zaGFkb3c6IDBweCAycHggNHB4IDBweCByZ2JhKDAsMCwwLDAuMzApO1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX21pZHRvbmU7XG59XG5cbi8qIEBrZXlmcmFtZXMgcHVsc2Uge1xuICAwJSB7XG4gICAgb3BhY2l0eTogMC4yO1xuICB9XG4gIDUwJSB7XG4gICAgb3BhY2l0eTogMTtcbiAgfVxuICAxMDAlIHtcbiAgICBvcGFjaXR5OiAwLjI7XG4gIH1cbn1cbkAtd2Via2l0LWtleWZyYW1lcyBwdWxzZSB7XG4gIDAlIHtcbiAgICBvcGFjaXR5OiAwLjI7XG4gIH1cbiAgNTAlIHtcbiAgICBvcGFjaXR5OiAxO1xuICB9XG4gIDEwMCUge1xuICAgIG9wYWNpdHk6IDAuMjtcbiAgfVxufVxuQC1tb3ota2V5ZnJhbWVzIHB1bHNlIHtcbiAgMCUge1xuICAgIG9wYWNpdHk6IDAuMjtcbiAgfVxuICA1MCUge1xuICAgIG9wYWNpdHk6IDE7XG4gIH1cbiAgMTAwJSB7XG4gICAgb3BhY2l0eTogMC4yO1xuICB9XG59XG5ALW8ta2V5ZnJhbWVzIHB1bHNlIHtcbiAgMCUge1xuICAgIG9wYWNpdHk6IDAuMjtcbiAgfVxuICA1MCUge1xuICAgIG9wYWNpdHk6IDE7XG4gIH1cbiAgMTAwJSB7XG4gICAgb3BhY2l0eTogMC4yO1xuICB9XG59ICovXG5cbi5tYXJrZXJTbWFydENpdGl6ZW5PZmZsaW5lIHtcbiAgYm9yZGVyOiAycHggc29saWQgcmdiYSgwLDAsMCwgMC4yKTtcbiAgYm9yZGVyLXJhZGl1czogMjBweDtcbiAgYm94LXNoYWRvdzogMHB4IDJweCA0cHggMHB4IHJnYmEoMCwwLDAsMC4zMCk7XG4gIGJhY2tncm91bmQtY29sb3I6IHJnYmEoMCwwLDAsMC4yKTtcbn1cblxuLyogLm1hcmtlcl9ibGluayB7XG4gIC13ZWJraXQtYW5pbWF0aW9uOiBwdWxzZSAycyBsaW5lYXIgaW5maW5pdGU7XG4gIC1tb3otYW5pbWF0aW9uOiBwdWxzZSAycyBsaW5lYXIgaW5maW5pdGU7XG4gIC1tcy1hbmltYXRpb246IHB1bHNlIDJzIGxpbmVhciBpbmZpbml0ZTtcbiAgYW5pbWF0aW9uOiBwdWxzZSAycyBsaW5lYXIgaW5maW5pdGU7XG59ICovXG5cbi5wb3B1cCB7XG4gIHdpZHRoOiAxMDElOztcbiAgaGVpZ2h0OiAxMzVweDtcbiAgY29sb3I6ICRvZmZfYmxhY2s7XG4gIGZvbnQtZmFtaWx5OiAnUm9ib3RvIENvbmRlbnNlZCdcbn1cblxuLnBvcHVwX3RvcCB7XG4gIGhlaWdodDogNzJweDtcbiAgYm9yZGVyLXJhZGl1czogMnB4IDJweCAwIDA7XG4gIGJhY2tncm91bmQtY29sb3I6ICRiYWNrZ3JvdW5kX2JsdWVfbGlnaHQ7XG4gIHBhZGRpbmctdG9wOjVweDtcblxuICBhIHtcbiAgICBjb2xvcjogJG9mZl9ibGFjaztcbiAgICAmOmhvdmVyIHtcbiAgICAgIGNvbG9yOiAkZ3JleV9kYXJrZXI7XG4gICAgfVxuICB9XG5cbiAgLnBvcHVwX25hbWUge1xuICAgIHBhZGRpbmc6IDAgMCAxcHggOHB4O1xuICAgIGZvbnQtc2l6ZTogMTZweDtcbiAgICBmb250LXdlaWdodDogYm9sZDtcbiAgfVxuICAucG9wdXBfdHlwZSB7XG4gICAgZm9udC1zaXplOiAxNHB4O1xuICAgIGxpbmUtaGVpZ2h0OiAxZW07XG4gICAgcGFkZGluZzogMXB4IDAgM3B4IDhweDtcbiAgfVxuICAucG9wdXBfdGltZSB7XG4gICAgZm9udC1zaXplOiAxMXB4O1xuICAgIHBhZGRpbmc6IDJweCAwIDAgOHB4O1xuXG4gICAgbWQtaWNvbiB7XG4gICAgICB3aWR0aDogMTBweDtcbiAgICAgIGhlaWdodDogMTBweDtcbiAgICAgIG1hcmdpbi1yaWdodDogNXB4O1xuICAgIH1cbiAgfVxufVxuXG4ucG9wdXBfYm90dG9tIHtcbiAgaGVpZ2h0OiA2OHB4O1xuXG4gIC5wb3B1cF9sb2NhdGlvbiB7XG4gICAgcGFkZGluZzogNXB4IDAgMCA4cHg7XG4gICAgZm9udC1zaXplOiAxNHB4O1xuXG4gICAgbWQtaWNvbiB7XG4gICAgICB3aWR0aDogOC40cHg7XG4gICAgICBoZWlnaHQ6IDEycHg7XG4gICAgICBtYXJnaW4tcmlnaHQ6IDRweDtcbiAgICB9XG4gIH1cbiAgLnBvcHVwX2xhYmVscyB7XG4gICAgcGFkZGluZzogOHB4IDAgMCA4cHg7XG4gICAgZm9udC1zaXplOiAxMnB4O1xuXG4gICAgLy8gc3BhbiB7XG4gICAgLy8gICAvLyBNYWtlIGJhbGxvb25zIG9uIG1hcCBzbWFsbGVyXG4gICAgLy8gICAvL2JvcmRlci1yYWRpdXM6IDIwcHg7XG4gICAgLy8gICAvL21hcmdpbi1yaWdodDogMTBweDtcbiAgICAvLyAgIC8vcGFkZGluZzogM3B4IDhweDtcbiAgICAvLyB9XG5cbiAgICB0YWcsIHNwYW4ge1xuICAgICAgbWFyZ2luLWJvdHRvbTogOHB4O1xuICAgICAgZmxvYXQ6IGxlZnQ7XG4gICAgfVxuICB9XG59XG4ucG9wdXBfaWNvbiBzdmc6bnRoLWNoaWxkKDIpIHtcbiAgd2lkdGg6IDA7XG4gIGhlaWdodDogMDtcbn1cblxuLyogS2l0IHR5cGUgY29sb3JzIGtpdFV0aWxzLnNlcnZpY2UuanMgOjogY2xhc3NpZnkoKSovXG5cbi5zY2sge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeWVsbG93O1xufVxuLnVua25vd24ge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiBncmV5O1xufVxuLyoqXG4gKiBPVkVSUklERSBMRUFGTEVUXG4gKi9cblxuLmxlYWZsZXQtbGVmdCB7XG4gIC8vIFRoaXMgaXMgYSBoYWNrLiBUaGUgcmlnaHQgd2F5IGlzIHRvIGNoYW5nZSB0aGUgbGVhZmxldCBBUEkuIFNlZSB0aGUgZG9jczpcbiAgLy8gaHR0cDovL2xlYWZsZXRqcy5jb20vcmVmZXJlbmNlLTAuNy43Lmh0bWwjY29udHJvbC1wb3NpdGlvblxuICBsZWZ0OiB1bnNldCAhaW1wb3J0YW50O1xuICByaWdodDogMTFweDtcbiAgdG9wOiAxNXB4ICFpbXBvcnRhbnQ7XG59XG5cbi5sZWFmbGV0LXBvcHVwLWNvbnRlbnQtd3JhcHBlciB7XG4gIGJvcmRlci1yYWRpdXM6IDRweDtcbiAgd2lkdGg6IGF1dG8gIWltcG9ydGFudDtcbiAgb3ZlcmZsb3c6IGhpZGRlbjtcbn1cblxuLmxlYWZsZXQtcG9wdXAtY29udGVudCB7XG4gIG1hcmdpbjogMDtcbiAgbWluLXdpZHRoOiAyNDVweDtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xuICB0b3A6IC0xcHg7XG4gIGxlZnQ6IC0xcHg7XG5cbiAgcCB7XG4gICAgbWFyZ2luOiAwO1xuICB9XG59XG5cbi5sZWFmbGV0LXBvcHVwLWNsb3NlLWJ1dHRvbiB7XG4gIGRpc3BsYXk6IG5vbmU7XG59XG5cbi5sZWFmbGV0LW92ZXJsYXktcGFuZSAubGVhZmxldC16b29tLWFuaW1hdGVkIHBhdGgge1xuICBzdHJva2U6IGdyZXk7XG59XG4vLyBvdmVycmlkZSB6LWluZGV4IG9mIG1hcCBjb250cm9sc1xuLmxlYWZsZXQtdG9wLmxlYWZsZXQtbGVmdCB7XG4gIHotaW5kZXg6IDE7XG59XG4ubGVhZmxldC1ib3R0b20ubGVhZmxldC1yaWdodCB7XG4gIHotaW5kZXg6IDE7XG59XG5cbi5tYXJrZXItY2x1c3RlciB7XG4gIGJveC1zaGFkb3c6IDBweCAycHggNHB4IDBweCByZ2JhKDAsMCwwLDAuMjApO1xuICBkaXYge1xuICAgIC8vbWFyZ2luLXRvcDogLTJweDtcbiAgICAvL21hcmdpbi1sZWZ0OiAtMnB4O1xuICAgIC8vd2lkdGg6IDQwcHg7XG4gICAgLy9oZWlnaHQ6IDQwcHg7XG4gICAgZm9udC1mYW1pbHk6ICdSb2JvdG8gQ29uZGVuc2VkJztcbiAgICBmb250LXNpemU6IDE0cHg7XG4gICAgLy9saW5lLWhlaWdodDogNDBweDtcbiAgICAvL2ZvbnQtd2VpZ2h0OiBub3JtYWw7XG4gIH1cbn1cblxuLm1hcmtlci1jbHVzdGVyLXNtYWxsLCAubWFya2VyLWNsdXN0ZXItbWVkaXVtLCAubWFya2VyLWNsdXN0ZXItbGFyZ2Uge1xuICAgIGJvcmRlci1yYWRpdXM6IDUwJTtcbiAgICBib3JkZXI6IDJweCBzb2xpZCByZ2JhKCR5ZWxsb3csIDAuOCk7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogcmdiYSgkeWVsbG93LCAwLjMpO1xuICBkaXYge1xuICAgIGJhY2tncm91bmQtY29sb3I6IHJnYmEoMCwyNTUsMTY4LDApO1xuICB9XG59XG5cbi8qXG4gIFBJQ0tFUiBDT01QT05FTlRcbiovXG5cbi5waWNrZXIge1xuICBjb2xvcjogJG9mZl9ibGFjaztcbn1cblxuLnBpY2tlcl9fd2Vla2RheSB7XG4gIGNvbG9yOiAkZ3JleV9kYXJrZXI7XG59XG5cbi5waWNrZXJfX25hdi0tcHJldjo6YmVmb3JlIHtcbiAgYm9yZGVyLXJpZ2h0OiAwLjc1ZW0gc29saWQgJGdyZXlfZGFya2VyO1xufVxuXG4ucGlja2VyX19uYXYtLW5leHQ6OmJlZm9yZSB7XG4gIGJvcmRlci1sZWZ0OiAwLjc1ZW0gc29saWQgJGdyZXlfZGFya2VyO1xufVxuXG4ucGlja2VyX19idXR0b24tLXRvZGF5OjpiZWZvcmUge1xuICAgIGJvcmRlci10b3A6IDAuNjZlbSBzb2xpZCAkYmx1ZTtcbn1cblxuLnBpY2tlcl9faG9sZGVyLnBpY2tlcl9jb250YWluZXIge1xuICBiYWNrZ3JvdW5kOiBub25lO1xuICB0cmFuc2l0aW9uOiBub25lO1xufVxuXG4ucGlja2VyX19idXR0b24tLWNsZWFyOjpiZWZvcmUge1xuICAgIGJvcmRlci10b3A6IDNweCBzb2xpZCAkcmVkO1xufVxuLnBpY2tlcl9fYnV0dG9uLS1jbG9zZSB7XG4gIGNvbG9yOiAkb2ZmX2JsYWNrO1xufVxuLnBpY2tlcl9fYnV0dG9uLS10b2RheSwgLnBpY2tlcl9fYnV0dG9uLS1jbGVhciB7XG4gIGNvbG9yOiAkb2ZmX2JsYWNrO1xufVxuLnBpY2tlcl9fZGF5LS1oaWdobGlnaHRlZCwgLnBpY2tlcl9fZGF5LS1zZWxlY3RlZCwgLnBpY2tlcl9fZGF5LS1zZWxlY3RlZDpob3ZlciwgLnBpY2tlci0tZm9jdXNlZCAucGlja2VyX19kYXktLXNlbGVjdGVke1xuICBiYWNrZ3JvdW5kOiAkeWVsbG93O1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeWVsbG93O1xuICBib3JkZXItY29sb3I6ICR5ZWxsb3c7XG4gIGNvbG9yOiAkb2ZmX2JsYWNrO1xufVxuLnBpY2tlcl9fZGF5LS1pbmZvY3VzOmhvdmVyLCAucGlja2VyX19uYXYtLXByZXY6aG92ZXIsIC5waWNrZXJfX25hdi0tbmV4dDpob3ZlciwgLnBpY2tlcl9fYnV0dG9uLS10b2RheTpob3ZlciwgLnBpY2tlcl9fYnV0dG9uLS1jbGVhcjpob3ZlciwgLnBpY2tlcl9fYnV0dG9uLS1jbG9zZTpob3ZlciB7XG4gIGJhY2tncm91bmQ6ICR5ZWxsb3dfbGlnaHRlcjtcbiAgYm9yZGVyLWNvbG9yOiAkeWVsbG93X2xpZ2h0ZXI7XG59XG5cbi5kYXRlX3BpY2tlciB7XG4gIGhlaWdodDogMjRweDtcbiAgYm9yZGVyLXJhZGl1czogMnB4O1xuICBib3JkZXI6IDFweCBzb2xpZCAkZ3JleV9kYXJrZXI7XG4gIGNvbG9yOiAkZ3JleV9kYXJrZXI7XG4gIGJhY2tncm91bmQtY29sb3I6IHRyYW5zcGFyZW50O1xufVxuXG4uc2VsZWN0X2ltYWdlIHtcbiAgbWFyZ2luLXJpZ2h0OiAxMHB4O1xufVxuXG4vKipcbiAqIERST1BET1dOIE1FTlVcbiAqIE92ZXJyaWRlIHN0eWxlIGZvciBkcm9wZG93biBtZW51IGZyb20gYW5ndWxhci1kcm9wZG93bnMgZGVwZW5kZW5jeVxuICovXG4gLndyYXAtZGQtbWVudSAuZHJvcGRvd24gbGkgYSB7XG4gICBjb2xvcjogJHNlY29uZGFyeV9jb2xvcjtcbiB9XG5cbi5raXRfbWVudSB7XG5cbiAgLndyYXAtZGQtbWVudSB7XG4gICAgd2lkdGg6IDMwcHg7XG4gICAgZGlzcGxheTogaW5saW5lLXRhYmxlO1xuICAgIHBhZGRpbmc6IDAgMDtcbiAgfVxuICAud3JhcC1kZC1tZW51IC5kcm9wZG93biB7XG4gICAgdG9wOiA3MCU7XG4gICAgbGluZS1oZWlnaHQ6IDFlbTtcbiAgICBsZWZ0OiAtMjVweDtcbiAgICByaWdodDogLTI1cHg7XG4gICAgbGkgYSB7XG4gICAgICB0ZXh0LWFsaWduOiBjZW50ZXI7XG4gICAgICBib3JkZXItYm90dG9tOiAwcHg7XG4gICAgfVxuICB9XG59XG4ua2l0TGlzdF9wYXJlbnQge1xuXG4gIC53cmFwLWRkLW1lbnUge1xuICAgIHdpZHRoOiA1MHB4O1xuICAgIGRpc3BsYXk6IGlubGluZS1ibG9jaztcbiAgfVxuICAud3JhcC1kZC1tZW51IC5kcm9wZG93biB7XG4gICAgdG9wOiA3MCU7XG4gICAgbGluZS1oZWlnaHQ6IDFlbTtcbiAgICBsZWZ0OiAtMjVweDtcbiAgICByaWdodDogLTI1cHg7XG4gICAgYm9yZGVyLXJhZGl1czogMHB4O1xuICAgIGxpIGEge1xuICAgICAgdGV4dC1hbGlnbjogY2VudGVyO1xuICAgICAgYm9yZGVyLWJvdHRvbTogMHB4O1xuICAgIH1cbiAgfVxufVxuXG4ua2l0X293bmVyX2Ryb3Bkb3duX2J1dHRvbiB7XG4gIGRpc3BsYXk6IGJsb2NrO1xuICBtYXJnaW46IDAgYXV0bztcbn1cblxuLyoqXG4gKiAgS0lUIExJU1RcbiAqXG4gKi9cbi5raXRMaXN0IHtcbiAgbWFyZ2luOiAwcHg7XG4gIGNvbG9yOiAkZ3JleTtcbiAgYm9yZGVyLXRvcDogMXB4IHNvbGlkICNFQUVDRjE7XG4gIGJvcmRlci1yaWdodDogMXB4IHNvbGlkICNFQUVDRjE7XG4gIGJvcmRlci1sZWZ0OiAxcHggc29saWQgI0VBRUNGMTtcbiAgYmFja2dyb3VuZC1jb2xvcjogd2hpdGU7XG4gIGJvcmRlci1yYWRpdXM6IDAgIWltcG9ydGFudDtcbiAgdGV4dC10cmFuc2Zvcm06IG5vbmU7XG4gIHBhZGRpbmc6MTVweDtcblxuICBoMSxoMixoMyxoNHtcbiAgICBsaW5lLWhlaWdodDogaW5oZXJpdDtcbiAgfVxuICBwe1xuICAgIGxpbmUtaGVpZ2h0OiAwLjRlbTtcbiAgfVxuXG4gIC53cmFwLWRkLW1lbnUge1xuICAgIHdpZHRoOiA1MHB4O1xuICAgIGRpc3BsYXk6IGlubGluZS1ibG9jaztcbiAgfVxuICAud3JhcC1kZC1tZW51IC5kcm9wZG93biB7XG4gICAgdG9wOiA3MCU7XG4gICAgbGluZS1oZWlnaHQ6IDFlbTtcbiAgICBsZWZ0OiAtMjVweDtcbiAgICByaWdodDogLTI1cHg7XG4gICAgYm9yZGVyLXJhZGl1czogMHB4O1xuICAgIGxpIGEge1xuICAgICAgdGV4dC1hbGlnbjogY2VudGVyO1xuICAgICAgYm9yZGVyLWJvdHRvbTogMHB4O1xuICAgIH1cbiAgfVxuXG4gIC5jaGlwcyB7XG4gICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgIHJpZ2h0OiAxMzBweDtcbiAgfVxuXG59XG5cbi5raXRMaXN0X2F2YXRhciB7XG4gIHdpZHRoOiA0NHB4O1xuICBoZWlnaHQ6IDQ0cHg7XG4gIGJvcmRlci1yYWRpdXM6IDIycHg7XG59XG4ua2l0TGlzdF9uYW1lIHtcbiAgbWFyZ2luLWJvdHRvbTogNXB4O1xuICBmb250LXNpemU6IDE4cHg7XG4gIGNvbG9yOiAkYmx1ZTtcbn1cbi5raXRMaXN0X3JpZ2h0IHtcbiAgcG9zaXRpb246IGFic29sdXRlO1xuICByaWdodDogMDtcblxuICAuY2hpcCAuY2hpcF9uYW1lIHtcbiAgICB0ZXh0LXRyYW5zZm9ybTogdXBwZXJjYXNlO1xuICB9XG59XG4ua2l0TGlzdF9zdGF0ZSB7XG4gIGNvbG9yOndoaXRlO1xuICB0ZXh0LXRyYW5zZm9ybTogdXBwZXJjYXNlO1xufVxuLmtpdExpc3Rfc3RhdGVfaGFzX3B1Ymxpc2hlZCB7XG4gIGNvbG9yOiAkZ3JlZW47XG59XG4ua2l0TGlzdF9zdGF0ZV9uZXZlcl9wdWJsaXNoZWQge1xuICBjb2xvcjogJHllbGxvdztcbn1cbi5raXRMaXN0X3N0YXRlX25vdF9jb25maWd1cmVkIHtcbiAgY29sb3I6ICNGNDNENEQ7XG59XG4ua2l0TGlzdF9kcm9wZG93bkJ1dHRvbiB7XG4gIGRpc3BsYXk6IGJsb2NrO1xuICBtYXJnaW46IDAgYXV0bztcblxuICBpbWcge1xuICAgIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgfVxufVxuXG4ua2l0TGlzdF9ub0tpdHMge1xuICBsaW5lLWhlaWdodDogNTtcbn1cblxuLmRyb3Bkb3duLWl0ZW0tYnV0dG9uIHtcbiAgICBib3JkZXItcmFkaXVzOiA1cHg7XG4gICAgdGV4dC1hbGlnbjogY2VudGVyO1xuICAgIHdpZHRoOiAxMDAlO1xufVxuLmRyb3Bkb3duLWl0ZW0tYnV0dG9uOmhvdmVye1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeWVsbG93ICFpbXBvcnRhbnRcbn1cblxubWQtY2FyZC5raXQtbGlzdC1pdGVtIHtcbiAgYm9yZGVyLXJhZGl1czogMTBweDtcbn1cblxuLyoqXG4gKiAgVVNFUiBQUk9GSUxFIFNUQVRFXG4gKiAgbW9zdCBzdHlsZXMgYXJlIHJlLXVzZWQgb24gbXkgcHJvZmlsZSBzdGF0ZVxuICovXG5cbi8qIHByb2ZpbGUgaGVhZGVyICovXG4ucHJvZmlsZV9oZWFkZXIge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2xpZ2h0ZXJfbWVudTtcbiAgY29sb3I6ICRmb250X2NvbG9yX2JsdWVfZ3JleTtcbiAgaGVpZ2h0OiAxOTJweDtcbiAgbWFyZ2luLXRvcDogNjRweDtcbn1cbi5wcm9maWxlX2hlYWRlcl9hdmF0YXIge1xuICB3aWR0aDogMTAwcHg7XG4gIGhlaWdodDogMTAwcHg7XG4gIGJvcmRlci1yYWRpdXM6IDUwcHg7XG4gIG1hcmdpbi1yaWdodDogMjlweDtcbiAgbWFyZ2luLWxlZnQ6IDI3cHg7XG59XG4ucHJvZmlsZV9oZWFkZXJfY29udGVudCB7XG4gID5kaXYge1xuICAgIG1hcmdpbjogNXB4IDA7XG4gIH1cbiAgc3ZnIHtcbiAgICBmaWxsOiAkZ3JleV9saWdodGVyO1xuICB9XG59XG4ucHJvZmlsZV9oZWFkZXJfY29udGVudF9hdmF0YXIge1xuICB3aWR0aDogMTJweDtcbiAgaGVpZ2h0OiAxMnB4O1xuICBtYXJnaW4tcmlnaHQ6IDZweDtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xuICB0b3A6IC0xcHg7XG59XG5cbi5wcm9maWxlX2NvbnRlbnQge1xuICBtaW4taGVpZ2h0OiA3MDBweDtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xufVxuLyogcHJvZmlsZSBzaWRlYmFyICovXG4ucHJvZmlsZV9zaWRlYmFyIHtcbiAgLy9taW4taGVpZ2h0OiA0MDBweDtcbiAgLy8gbWluLWhlaWdodDogaW5oZXJpdDtcbiAgLy8gd2lkdGg6IDI1NnB4O1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAjRjVGNUY1O1xuICBtaW4td2lkdGg6MjAwcHg7XG59XG4ucHJvZmlsZV9zaWRlYmFyX3RpdGxlIHtcbiAgY29sb3I6ICRvZmZfYmxhY2s7XG4gIGZvbnQtd2VpZ2h0OiBib2xkO1xufVxuLnByb2ZpbGVfc2lkZWJhcl9kZXNjcmlwdGlvbiB7XG4gIGZvbnQtc2l6ZTogMTRweDtcbn1cblxuLnByb2ZpbGVfc2lkZWJhcl9idXR0b24ge1xuICBoZWlnaHQ6IDY0cHg7XG4gIGNvbG9yOiAkdGVyY2lhcnlfY29sb3I7XG4gIHN2ZyB7IGZpbGw6ICRibHVlOyB9XG59XG5cbi5wcm9maWxlX3NpZGViYXJfYXZhdGFyIHtcbiAgd2lkdGg6IDEycHg7XG4gIGhlaWdodDogMTJweDtcbiAgbWFyZ2luLXJpZ2h0OiA4cHg7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgdG9wOiAtMnB4O1xufVxuXG4vKiBwcm9maWxlIG1haW4gY29udGVudCAqL1xuLnByb2ZpbGVfY29udGVudF9tYWluX3RvcCB7XG4gIG1hcmdpbjogMCAwIDQycHggMjBweDtcbn1cblxuLyoqXG4gKiAgTVkgUFJPRklMRSBTVEFURVxuICovXG5cbi8vb3ZlcnJpZGUgYW5ndWxhci1tYXRlcmlhbFxuLm15UHJvZmlsZV9zdGF0ZSB7XG4gIG1kLWluay1iYXJ7XG4gICAgLy8gVGFicyB1bmRlcmxpbmUgY29sb3IgY3lhblxuICAgIGJhY2tncm91bmQtY29sb3I6ICR0ZXJjaWFyeV9jb2xvcjtcbiAgfVxuICBtZC10YWIgbWQtdGFiLWxhYmVsIHtcbiAgICBvdmVyZmxvdzogdmlzaWJsZTtcbiAgfVxuICBtZC10YWJzIG1kLXRhYiB7XG4gICAgY29sb3I6ICRmb250X2NvbG9yX2JsdWVfZ3JleTtcbiAgfVxuICBtZC10YWJzIG1kLXRhYi5hY3RpdmUge1xuICAgIGNvbG9yOiB3aGl0ZTtcbiAgfVxuICBtZC10YWJzIG1kLXRhYnMtaW5rLWJhciB7XG4gICAgY29sb3I6ICRibHVlO1xuICAgIGJhY2tncm91bmQtY29sb3I6ICRibHVlO1xuICB9XG59XG5cbm1kLXRhYnMtY2FudmFzIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogIzJFMkUyRTtcbn1cblxuLm15UHJvZmlsZV9oZWFkZXIge1xuICBoZWlnaHQ6IDE4OHB4O1xuICBwb3NpdGlvbjogcmVsYXRpdmU7XG59XG4ubXlQcm9maWxlX2hlYWRlcl9jb250YWluZXIge1xuICBwYWRkaW5nLXRvcDogNTFweDtcbiAgbWFyZ2luLWxlZnQ6IDQ2cHg7XG59XG4ubXlQcm9maWxlX2hlYWRlcl9hdmF0YXIge1xuICBtYXJnaW4tbGVmdDogMDtcbn1cbi5teVByb2ZpbGVfdGFic19wYXJlbnQge1xuICBtZC10YWJzLXdyYXBwZXJ7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogJGJhY2tncm91bmRfYmx1ZV9kYXJrX21lbnU7XG4gIH1cbn1cbi8vIC5teVByb2ZpbGVfdGFicyB7XG4vLyB9XG4ubXlQcm9maWxlX3RhYl9pY29uIHtcbiAgd2lkdGg6IDE2cHg7XG4gIGhlaWdodDogMTZweDtcbiAgbWFyZ2luLXJpZ2h0OiA4cHg7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgdG9wOiAtMXB4O1xuICBzdmcge1xuICAgIGZpbGw6IHdoaXRlO1xuICB9XG59XG4ubXlQcm9maWxlX3NpZGViYXJfYnV0dG9uIHtcbiAgcGFkZGluZy1sZWZ0OiAyOHB4O1xufVxuLm15UHJvZmlsZV9jb250ZW50X2Zvcm0ge1xuICBtYXJnaW4tbGVmdDogMTEzcHg7XG4gIG1heC13aWR0aDogNTAwcHg7XG59XG4ubXlQcm9maWxlX2NvbnRlbnRfZm9ybV9pbnB1dCB7XG4gIG1hcmdpbjogMTZweCAwO1xuICBwb3NpdGlvbjogcmVsYXRpdmU7XG59XG4uY291bnRyeUlucHV0X2NvbnRhaW5lciB7XG4gIG1kLWF1dG9jb21wbGV0ZSB7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogdHJhbnNwYXJlbnQ7XG4gIH1cbiAgbWQtYXV0b2NvbXBsZXRlLXdyYXAge1xuICAgIGJveC1zaGFkb3c6IG5vbmUgIWltcG9ydGFudDtcbiAgfVxuICBpbnB1dCB7XG4gICAgZm9udC1zaXplOiAxMDAlO1xuICB9XG4gIGxhYmVsIHtcbiAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gICAgYm90dG9tOiA1NXB4O1xuICB9XG59XG4ubXlQcm9maWxlX2Zvcm1fYXZhdGFyIHtcbiAgbWFyZ2luLWJvdHRvbTogMzBweDtcbn1cbi5teVByb2ZpbGVfZm9ybV9hdmF0YXJJbWFnZSB7XG4gIHdpZHRoOiA2NHB4O1xuICBoZWlnaHQ6IDY0cHg7XG4gIGJvcmRlci1yYWRpdXM6IDMycHg7XG4gIG1hcmdpbi1yaWdodDogNyU7XG59XG4ubXlQcm9maWxlX2FwaUtleV90ZXh0IHtcbiAgY29sb3I6ICRmb250X2NvbG9yX2JsdWVfZ3JleTtcbiAgbWFyZ2luLXJpZ2h0OiA1cHg7XG4gIHdpZHRoOiAxMjBweDtcbn1cbi5teVByb2ZpbGVfYXBpS2V5X251bWJlciB7XG4gIGNvbG9yOiAjOUQ5RDlFO1xuICBmb250LXdlaWdodDogYm9sZDtcbiAgYmFja2dyb3VuZC1jb2xvcjogI0YyRjJGMjtcbiAgcGFkZGluZzogNXB4IDEwcHg7XG59XG4ubXlQcm9maWxlX2FwaUtleV9yZWZyZXNoQnV0dG9uIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJGZvbnRfY29sb3JfYmx1ZV9ncmV5O1xuICBoZWlnaHQ6IDI5cHg7XG5cbiAgbWQtaWNvbiB7XG4gICAgaGVpZ2h0OiAxNXB4O1xuICAgIHdpZHRoOiAxNXB4O1xuICAgIGRpc3BsYXk6IGJsb2NrO1xuICAgIG1hcmdpbjogM3B4IGF1dG87XG4gIH1cbn1cbi5tZC1idXR0b24ubXlQcm9maWxlX2FwaUtleV9yZWZyZXNoQnV0dG9uIHtcbiAgYm9yZGVyLXJhZGl1czogMCA0cHggNHB4IDA7XG59XG4ubWQtYnV0dG9uLm15UHJvZmlsZV9hcGlLZXlfcmVmcmVzaEJ1dHRvbjphY3RpdmUsIC5tZC1idXR0b24ubXlQcm9maWxlX2FwaUtleV9yZWZyZXNoQnV0dG9uOmhvdmVyLCAubWQtYnV0dG9uLm15UHJvZmlsZV9hcGlLZXlfcmVmcmVzaEJ1dHRvbjpmb2N1cyB7XG4gIGJhY2tncm91bmQtY29sb3I6ICRmb250X2NvbG9yX2JsdWVfZ3JleTtcbn1cbi5teVByb2ZpbGVfZm9ybV9yZW1vdmVCdXR0b24ge1xuICBkaXNwbGF5OiBibG9jaztcbiAgY29sb3I6ICRhbGVydF9yZWQ7XG4gIG1hcmdpbi1ib3R0b206IDVweDtcbn1cbi5teVByb2ZpbGVfdXBkYXRlRm9ybV9lcnJvciB7XG4gIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgdG9wOiA1MHB4O1xuICBsZWZ0OiAycHg7XG4gIGNvbG9yOiByZWQ7XG4gIGZvbnQtc2l6ZTogMTJweDtcbiAgbGluZS1oZWlnaHQ6IDE7XG59XG4ubXlQcm9maWxlX2FwaUtleV9ibG9jayB7XG4gIG1hcmdpbi10b3A6IDYzcHg7XG4gIG1hcmdpbi1ib3R0b206IDY1cHg7XG59XG5cbi5teVByb2ZpbGVfYXBpS2V5IHtcbiAgbWFyZ2luLXRvcDogMTBweDtcbiAgbWFyZ2luLWJvdHRvbTogMTBweDtcbiAgc3ZnIHtcbiAgICBmaWxsOiAkZ3JleV9kYXJrZXI7XG4gIH1cbn1cblxuLm1kLWJ1dHRvbi5teVByb2ZpbGVfYWRkS2l0QnV0dG9uOmZvY3VzIHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJGJsdWU7XG59XG4ubWQtYnV0dG9uLm15UHJvZmlsZV9hZGRLaXRCdXR0b246aG92ZXIge1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmx1ZTtcbn1cbi5teVByb2ZpbGVfaGlkZGVuaHJlZntcbiAgcG9zaXRpb246YWJzb2x1dGU7XG4gIGhlaWdodDogMTAwJTtcbiAgd2lkdGg6MTAwJTtcbiAgdG9wOjA7XG4gIGxlZnQ6MDtcbn1cblxuLyoqXG4gKiBUT09MIExJU1RcbiAqXG4gKi9cblxuIGEua2l0TGlzdCB7XG4gICBoNCB7XG4gICAgIGNvbG9yOiAkb2ZmX2JsYWNrO1xuICAgfVxuICAgcCB7XG4gICAgIGNvbG9yOiAkZ3JleV9kYXJrZXI7XG4gICB9XG4gfVxuXG4uY2xlYXJmaXgge1xuICBjbGVhcjogYm90aDtcbn1cblxuLypcbiAgRk9PVEVSIENPTVBPTkVOVFxuKi9cblxuZm9vdGVyIHtcbi8vICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHotaW5kZXg6IDE7XG4gIHdpZHRoOiAxMDAlO1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2RhcmtfYmFja2dyb3VuZDtcbn1cblxuLmZvb3Rlcl9pY29uIHtcbiAgZGlzcGxheTogYmxvY2s7XG4gIHdpZHRoOiAzMnB4O1xuICBoZWlnaHQ6IDMycHg7XG59XG4uZmxhZyB7XG4gIHBhZGRpbmctcmlnaHQ6IDE1cHg7XG59XG4uYm9yZGVyLXdoaXRlLmZvb3Rlci1ibG9ja3tcbiAgYm9yZGVyOiAycHggd2hpdGUgc29saWQ7XG59XG4udXB0aW1lcm9ib3QtbG9nbyB7XG4gIHdpZHRoOiAxNTBweDtcbiAganVzdGlmeS1zZWxmOiByaWdodDtcbiAgdmVydGljYWwtYWxpZ246IGNlbnRlcjtcbiAgcGFkZGluZy10b3A6IDEwcHg7XG59XG4udXB0aW1lcm9ib3Qtc3BvbnNvciB7XG4gIGRpc3BsYXk6IGZsZXg7XG4gIGZsZXgtZGlyZWN0aW9uOiBjb2x1bW47XG4gIGp1c3RpZnktY29udGVudDogc3RhcnQ7XG4gIGFsaWduLWl0ZW1zOiBmbGV4LXN0YXJ0O1xufVxuLnVwdGltZXJvYm90LXNwb25zb3ItdGV4dCBwIHtcbiAgZm9udC1zaXplOiAxNXB4ICFpbXBvcnRhbnQ7XG4gIG1hcmdpbjogMHB4O1xufVxuQG1lZGlhIG9ubHkgc2NyZWVuIGFuZCAobWF4LXdpZHRoOiA5NTBweCkge1xuICAuY29sb3Itd2hpdGV7XG4gICAgdGV4dC1hbGlnbjogY2VudGVyO1xuICB9XG4gIC5mbGFnIHtcbiAgICBwYWRkaW5nLXJpZ2h0OiAwcHg7XG4gIH1cbiAgLmZvb3Rlci1ibG9jayB7XG4gICAgbWFyZ2luLWJvdHRvbTogMjBweCAhaW1wb3J0YW50O1xuICB9XG4gIC5zcG9uc29yIHtcbiAgICBqdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjtcbiAgICBhbGlnbi1pdGVtczogY2VudGVyO1xuICB9XG59XG5AbWVkaWEgKG1heC13aWR0aDogNTk5cHgpIHtcbiAgLmxheW91dC1hbGlnbi14cy1zdGFydC1zdHJldGNoLCAubGF5b3V0LWFsaWduLXhzLWNlbnRlci1zdHJldGNoLCAubGF5b3V0LWFsaWduLXhzLWVuZC1zdHJldGNoLCAubGF5b3V0LWFsaWduLXhzLXNwYWNlLWJldHdlZW4tc3RyZXRjaCwgLmxheW91dC1hbGlnbi14cy1zcGFjZS1hcm91bmQtc3RyZXRjaHtcbiAgICBhbGlnbi1pdGVtczogY2VudGVyO1xuICB9XG59XG5cblxuLypcbiAgUkVDT1ZFUlkgUEFTU1dPUkQgU1RBVEVcbiovXG4ucmVjb3ZlcnlfaGVhZGVyIHtcbiAgd2lkdGg6IDEwMCU7XG4gIGhlaWdodDogMTI4cHg7XG4gIGJhY2tncm91bmQtY29sb3I6ICRiYWNrZ3JvdW5kX2JsdWVfZGFya19iYWNrZ3JvdW5kO1xufVxuXG4ucmVjb3ZlcnlfaGVhZGVySWNvbiB7XG4gIGRpc3BsYXk6IGJsb2NrO1xuICB3aWR0aDogNTBweDtcbiAgaGVpZ2h0OiA1MHB4O1xufVxuXG4vKlxuICBUSU1FTElORSBDT01QT05FTlRcblxuICB1c2VkIG9uIGFkZCBhbmQgc2V0dXAga2l0XG4qL1xuXG4udGltZWxpbmUge1xuICAvLyBtYXJnaW4tdG9wOiA2NHB4O1xuICAvLyBiYWNrZ3JvdW5kLWNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2RhcmtfbWVudTtcbiAgaGVpZ2h0OiAkdGltZWxpbmUtaGVpZ2h0O1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2xpZ2h0ZXJfbWVudTtcbiAgcG9zaXRpb246IGZpeGVkO1xuICB3aWR0aDogMTAwJTtcbiAgei1pbmRleDogMTA7XG4gIHRvcDogNjRweDtcbiAgYm94LXNoYWRvdzogMCA0cHggNXB4IC0ycHggJGJhY2tncm91bmRfYmx1ZV9saWdodGVyX21lbnU7XG4gIHBhZGRpbmc6IDMwcHg7XG59XG5cbkBtZWRpYSAobWF4LXdpZHRoOiA5NjBweCkge1xuICAudGltZWxpbmUtYnRuLWV4dHJhIHtcbiAgICBkaXNwbGF5OiBOb25lO1xuICB9XG59XG5cbkBtZWRpYSAobWluLXdpZHRoOiAwcHgpIGFuZCAobWF4LXdpZHRoOiA5NTlweCkgYW5kIChvcmllbnRhdGlvbjogcG9ydHJhaXQpIHtcbiAgLnRpbWVsaW5lIHtcbiAgICB0b3A6IDU2cHg7XG4gIH1cbiAgLnByb2ZpbGVfaGVhZGVyIHtcbiAgICBtYXJnaW4tdG9wOiA1M3B4O1xuICB9XG59XG5cbkBtZWRpYSAobWluLXdpZHRoOiAwcHgpIGFuZCAobWF4LXdpZHRoOiA5NTlweCkgYW5kIChvcmllbnRhdGlvbjogbGFuZHNjYXBlKSB7XG4gIC50aW1lbGluZSB7XG4gICAgdG9wOiA1M3B4O1xuICB9XG4gIC5wcm9maWxlX2hlYWRlciB7XG4gICAgbWFyZ2luLXRvcDogNTNweDtcbiAgfVxufVxuXG4udGltZWxpbmVfY29udGFpbmVyIHtcbiAgd2lkdGg6IDEwMCU7XG59XG4udGltZWxpbmVfbGluZSB7XG4gIHdpZHRoOiA1MDBweDtcbiAgYm9yZGVyOiAxcHggc29saWQgJHllbGxvdztcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xuICB0b3A6IDE0cHg7XG59XG4udGltZWxpbmVfbGluZV9zbWFsbCB7XG4gIHdpZHRoOiAxMDBweCAhaW1wb3J0YW50O1xuICBtYXJnaW46IDBweCAyMHB4O1xuICB0b3A6MHB4ICFpbXBvcnRhbnQ7XG59XG5cbi50aW1lbGluZS10aXRsZSB7XG4gIHRleHQtdHJhbnNmb3JtOiB1cHBlcmNhc2U7XG4gIGNvbG9yOiAkeWVsbG93O1xuICBmb250LXNpemU6MjRweDtcbiAgZm9udC13ZWlnaHQ6IDYwMDsgLy8gVE9ETzogQ29zbWV0aWNcblxuICAmLnZlcnRpY2FsIHtcbiAgICBtYXJnaW4tdG9wOiAxMHB4O1xuICB9XG59XG5cbi50aW1lbGluZV9zdGVwTmFtZSB7XG4gIHRleHQtdHJhbnNmb3JtOiB1cHBlcmNhc2U7XG4gIGNvbG9yOiAkeWVsbG93O1xuICBmb250LXNpemU6MThweDtcblxuICAmLnZlcnRpY2FsIHtcbiAgICBtYXJnaW4tdG9wOiAxMHB4O1xuICB9XG59XG5cbi50aW1lbGluZV9jb250ZW50IHtcbiAgbWluLWhlaWdodDogNDAwcHg7XG4gIHdpZHRoOiAxMDAlO1xuICBtYXJnaW4tdG9wOiAkY29udGVudC1tYXJnaW4tdG9wO1xufVxuXG4vLyAudGltZWxpbmVfYnV0dG9uIHtcbi8vICAgd2lkdGg6IDk5JTtcbi8vICAgaGVpZ2h0OiA5MHB4O1xuLy8gICBmb250LXNpemU6IDI2cHg7XG5cbi8vICAgJi50aW1lbGluZV9idXR0b25PcGVuLm1kLXByaW1hcnkge1xuLy8gICAgIGNvbG9yOiAkdGVyY2lhcnlfY29sb3I7XG4vLyAgICAgdGV4dC1hbGlnbjogY2VudGVyO1xuXG4vLyAgICAgJi5pbnZlcnRlZHtcbi8vICAgICAgIGNvbG9yOiB3aGl0ZTtcbi8vICAgICAgIGJhY2tncm91bmQtY29sb3I6ICR0ZXJjaWFyeV9jb2xvcjtcblxuLy8gICAgICAgc21hbGwge1xuLy8gICAgICAgICBjb2xvcjogd2hpdGU7XG4vLyAgICAgICB9XG4vLyAgICAgfVxuXG4vLyAgICAgLm1haW4ge1xuLy8gICAgICAgbWFyZ2luOiAxMHB4O1xuLy8gICAgICAgZm9udC1zaXplOiAzMnB4O1xuLy8gICAgIH1cblxuLy8gICAgIC5zdWIge1xuLy8gICAgICAgbWFyZ2luOiAxMHB4O1xuLy8gICAgICAgZm9udC1zaXplOiAxNnB4O1xuLy8gICAgIH1cblxuLy8gICAgIHNtYWxsIHtcbi8vICAgICAgIGNvbG9yOiByZ2JhKDAsMCwwLDAuNTQpO1xuLy8gICAgICAgdGV4dC10cmFuc2Zvcm06IGluaXRpYWw7XG4vLyAgICAgICBkaXNwbGF5OiBibG9jaztcbi8vICAgICAgIGZvbnQtc2l6ZTogMTZweDtcbi8vICAgICB9XG5cbi8vICAgfVxuLy8gfVxuXG4ubWQtYnV0dG9uLnRpbWVsaW5lX2J1dHRvbkJhY2t7XG4gIG1hcmdpbi1sZWZ0OiBhdXRvO1xuICBtYXJnaW4tcmlnaHQ6IDIwcHg7XG59XG5cbnNlY3Rpb24ucmVsYXhlZC1sYXlvdXQge1xuICBwYWRkaW5nOiAxMHB4IDI1cHg7XG59XG5cblxuXG4vKlxuICBBREQgS0lUIFNUQVRFXG4qL1xuXG4ua2l0X2RhdGFDaGFuZ2Uge1xuICAuZm9ybV9ibG9jayB7XG4gICAgcGFkZGluZzogNDBweCAzMHB4O1xuICAgIGJhY2tncm91bmQtY29sb3I6ICNGRkY7XG5cbiAgICAmLmlzRXZlbiB7XG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjRjlGOUZCO1xuICAgIH1cbiAgICAvLyAmLmZvcm1fYmxvY2tOb3JtYWwge1xuICAgIC8vICAvLyBtaW4taGVpZ2h0OiAyNTBweDtcbiAgICAvLyB9XG4gICAgLy8gJi5mb3JtX2Jsb2NrTWFwIHtcbiAgICAvLyAgIC8vaGVpZ2h0OiAyOTBweDtcbiAgICAvLyB9XG4gIH1cbiAgLmZvcm1fYmxvY2tJbnB1dCB7XG4gICAgaGVpZ2h0OiBpbmhlcml0O1xuICAgLy8gbWFyZ2luLWxlZnQ6IDE1MHB4O1xuICB9XG4gIC5mb3JtX2Jsb2NrSW5wdXRfY29udGFpbmVyIHtcbiAgICBoZWlnaHQ6IGluaGVyaXQ7XG4gIH1cbiAgLmZvcm1fYmxvY2tJbnB1dF9idXR0b24ge1xuICAgIGhlaWdodDppbmhlcml0O1xuICB9XG4gIC5mb3JtX2Jsb2NrSW5wdXRfbWFwIHtcbiAgICBoZWlnaHQ6IDI1MHB4O1xuICAgIG1pbi13aWR0aDogMjUwcHg7XG4gIH1cbiAgLmZvcm1fYmxvY2tJbnB1dF9zZWxlY3QgbGFiZWx7XG4gICAgY29sb3I6ICR0ZXJjaWFyeV9jb2xvcjtcbiAgICBtYXJnaW4tcmlnaHQ6IDEwcHg7XG4gIH1cbiAgLmZvcm1fYmxvY2tJbnB1dF9jaGlwcyB7XG4gICAgbWFyZ2luLXRvcDogMzVweDtcbiAgfVxuICAuZm9ybV9ibG9ja0NvbnRlbnQge1xuICAgIHBhZGRpbmctdG9wOiA1cHg7XG4gICAgZmxvYXQ6IGxlZnQ7XG4gICAgaW1nLCBkaXYge1xuICAgICAgZmxvYXQ6IGxlZnQ7XG4gICAgfVxuICAgIGltZyB7XG4gICAgICBib3JkZXItcmFkaXVzOiA1MCU7XG4gICAgICB3aWR0aDogMTIwcHg7XG4gICAgICBoZWlnaHQ6IDEyMHB4O1xuICAgICAgbWFyZ2luOiA1cHggMjBweDtcbiAgICB9XG4gIH1cbiAgLmZvcm1fYmxvY2tDb250ZW50X2ltYWdlIHtcbiAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gICAgdG9wOiAzcHg7XG4gIH1cbiAgLmZvcm1fYmxvY2tDb250ZW50X3RleHQge1xuICAgIG1hcmdpbi1sZWZ0OiAxMHB4O1xuICAgIHdpZHRoOiAyNDBweDtcbiAgICAmLmxvbmd7XG4gICAgICB3aWR0aDogYXV0byAhaW1wb3J0YW50O1xuICAgIH1cbiAgICBoMiB7XG4gICAgICBtYXJnaW4tdG9wOiAwO1xuICAgICAgbWFyZ2luLWJvdHRvbTogOHB4O1xuICAgIH1cbiAgICBwIHtcbiAgICAgIGZvbnQtc2l6ZTogMTRweDtcbiAgICAgIGNvbG9yOiAjQTVBNUE1O1xuICAgICAgbGluZS1oZWlnaHQ6IDEuNTtcbiAgICB9XG4gIH1cbiAgLmZvcm1fZXJyb3JzIHtcbiAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgYm90dG9tOiAxNSU7XG4gICAgbGVmdDogMDtcbiAgICBjb2xvcjogcmVkO1xuICAgIGZvbnQtc2l6ZTogMC43ZW07XG4gICAgbGluZS1oZWlnaHQ6IDE7XG4gIH1cbn1cblxuLmVtb2ppe1xuICBmb250LXdlaWdodDo0MDA7XG59XG5cbi8qXG4gIFNUQVRJQyBQQUdFU1xuKi9cbi5zdGF0aWNfcGFnZXtcbiAgLnRpbWVsaW5le1xuICAgIGFsaWduLWl0ZW1zOiBmbGV4LXN0YXJ0IHN0cmV0Y2g7XG5cbiAgICAuY29udGVudHtcbiAgICAgIHBhZGRpbmc6MCAwcHg7XG4gICAgfVxuICB9XG5cbiAgLmNvbnRlbnR7XG4gICAgbWF4LXdpZHRoOiA1MGVtO1xuICAgIG1hcmdpbjogMCBhdXRvO1xuICAgIHRleHQtYWxpZ246IGp1c3RpZnk7XG4gICAgcGFkZGluZzo0MHB4IDEwcHg7XG4gIH1cblxuICAuZnVsbC13aWR0aC1pbWcge1xuICAgIHdpZHRoOiAxMDAlO1xuICAgIGhlaWdodDogYXV0bztcbiAgICBtYXgtd2lkdGg6IDIwMDBweDtcbiAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gICAgbWFyZ2luOiAwIDAgLTVweCAwO1xuICB9XG5cbiAgLmVtYmVkLWNvbnRhaW5lciB7XG4gICAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gICAgICBwYWRkaW5nLWJvdHRvbTogNTYuMjUlO1xuICAgICAgaGVpZ2h0OiAwO1xuICAgICAgb3ZlcmZsb3c6IGhpZGRlbjtcbiAgICAgIG1heC13aWR0aDogMTAwJTtcbiAgICAgIGlmcmFtZSwgb2JqZWN0LCBlbWJlZCB7XG4gICAgICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICAgICAgdG9wOiAwO1xuICAgICAgICBsZWZ0OiAwO1xuICAgICAgICB3aWR0aDogMTAwJTtcbiAgICAgICAgaGVpZ2h0OiAxMDAlO1xuICAgICAgfVxuICB9XG5cbiAgLmNlbnRlciB7XG4gICAgICB0ZXh0LWFsaWduOiBjZW50ZXI7XG4gICAgICBtYXJnaW4tdG9wOiAzMHB4O1xuICB9XG5cbiAgLnJvd3tcbiAgICBtaW4taGVpZ2h0OiAyMDBweDtcblxuICAgICY6bnRoLWNoaWxkKG9kZCl7XG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjRjlGOUZCO1xuICAgIH1cbiAgfVxuICBoMXtcbiAgICBjb2xvcjp3aGl0ZTtcbiAgfVxuICBoNHtcbiAgICBmb250LXdlaWdodDogYm9sZDtcbiAgICBtYXJnaW4tYm90dG9tOiAwLjJlbTtcbiAgfVxuXG4gIC5tZC1idXR0b24ubWQtcHJpbWFyeS5tZC1yYWlzZWQsIC5tZC1idXR0b24ubWQtcHJpbWFyeS5tZC1mYWJ7XG4gICAgcGFkZGluZzogNnB4IDIwcHg7XG4gICAgaGVpZ2h0OiA0NHB4O1xuICAgIGJvcmRlci1yYWRpdXM6IDIycHg7XG4gICAgZm9udC1zaXplOiAxOHB4O1xuICAgIGJhY2tncm91bmQtY29sb3I6ICRibHVlO1xuICAgIGNvbG9yOndoaXRlO1xuXG4gICAgJjpob3ZlciwgJjpmb2N1c3tcbiAgICAgIGJhY2tncm91bmQtY29sb3I6ICMzQzk4QUY7XG4gICAgfVxuXG4gICAgJjpub3QoW2Rpc2FibGVkXSl7XG4gICAgICBib3gtc2hhZG93OiBub25lO1xuICAgIH1cbiAgfVxuXG4gIC5tZC1idXR0b24ubWQtcHJpbWFyeXtcbiAgICBjb2xvcjogJGJsdWU7XG4gIH1cbn1cblxuLm5vdC1mb3VuZC00MDQge1xuICBoZWlnaHQ6IDM2dmg7XG59XG5cbkBtZWRpYSAobWluLXdpZHRoOiA5NjBweCkge1xuICAubm90LWZvdW5kLTQwNCB7XG4gICAgaGVpZ2h0OiA2Ny4zdmg7XG4gIH1cbn1cblxuLy8gQWJvdXRcblxuLnNlbnNvci1pbWFnZS1tYXJnaW4ge1xuICBtYXJnaW4tdG9wOiA0MHB4O1xufVxuXG5oMy5uby1tYXJnaW4ge1xuICBtYXJnaW46IDA7XG59XG5cbnAubm8tbWFyZ2luIHtcbiAgbWFyZ2luOiAwO1xufVxuXG4udGVhbS1jZWxscy1tYXJnaW4ge1xuICBtYXJnaW4tYm90dG9tOiAyNXB4O1xufVxuXG4uc3VidGl0bGUtc2VwYXJhdGlvbiB7XG4gIG1hcmdpbi10b3A6IDYwcHg7XG59XG5cbi8vIEZlZWRiYWNrXG5cbi5kb29yYmVsbC1idXR0b24ge1xuIGJhY2tncm91bmQtY29sb3I6ICR5ZWxsb3cgIWltcG9ydGFudDtcbn1cblxuLy8gU3R5bGVndWlkZVxuXG4uc3R5bGVndWlkZSB7XG4gIC5pbmZvLXRleHQge1xuICAgIGNvbG9yOiByZ2JhKDAsIDAsIDAsIDAuMyk7XG4gIH1cblxuICAuaW5mby10ZXh0LWRhcmsge1xuICAgIGNvbG9yOiByZ2JhKDI1NSwgMjU1LCAyNTUsIDAuMykgIWltcG9ydGFudDtcbiAgfVxuXG4gIC5kYXJrLXRoZW1lLWRpdmlkZXIge1xuICAgIGJvcmRlci10b3AtY29sb3I6IHJnYmEoMjU1LDI1NSwyNTUsMC4xMik7XG4gIH1cblxuICBzbWFsbCB7XG4gICAgbWFyZ2luLXRvcDogMjBweDtcbiAgfVxuXG4gIC5kYXJrLXRleHQtc2VjdGlvbiB7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogJGJhY2tncm91bmRfYmx1ZV9kYXJrX2JhY2tncm91bmQ7XG5cbiAgICBoMSB7XG4gICAgICBmb250LXNpemU6IDEuOGVtO1xuICAgICAgY29sb3I6ICNmZmY7XG4gICAgfVxuXG4gICAgaDIge1xuICAgICAgY29sb3I6ICNmZmY7XG4gICAgfVxuXG4gICAgaDQge1xuICAgICAgY29sb3I6ICRmb250X2NvbG9yX2JsdWVfZ3JleTtcbiAgICB9XG5cbiAgICBoNiB7XG4gICAgICBjb2xvcjogIzg3Q0NERDtcbiAgICB9XG5cbiAgICBwe1xuICAgICAgY29sb3I6ICNDOEU2RUQ7XG4gICAgICBmb250LXdlaWdodDogMzAwO1xuICAgIH1cblxuICAgIHNtYWxsIHtcbiAgICAgIGNvbG9yOiAkc2Vjb25kYXJ5X2NvbG9yX2xpZ2h0O1xuICAgIH1cbiAgfVxuXG4gIC5zZWN0aW9uLXBhZGRpbmcge1xuICAgIHBhZGRpbmc6IDYwcHg7XG4gIH1cblxuICAuY29sb3JzLXNlY3Rpb24ge1xuICAgIHAge1xuICAgICAgcGFkZGluZy1sZWZ0OiAyMHB4O1xuICAgIH1cblxuICAgIC5zZWNvbmRhcnktY29sb3Ige1xuICAgICAgYmFja2dyb3VuZC1jb2xvcjogJHNlY29uZGFyeV9jb2xvcjtcbiAgICAgIHAge1xuICAgICAgICBjb2xvcjogI2ZmZjtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAudGVyY2lhcnlfY29sb3Ige1xuICAgICAgYmFja2dyb3VuZC1jb2xvcjogJHRlcmNpYXJ5X2NvbG9yO1xuICAgICAgcCB7XG4gICAgICAgIGNvbG9yOiAjZmZmO1xuICAgICAgfVxuICAgIH1cblxuICAgIC5zZWNvbmRhcnlfY29sb3JfbGlnaHQge1xuICAgICAgYmFja2dyb3VuZC1jb2xvcjogIzhEQjJCQTtcbiAgICAgIHAge1xuICAgICAgICBjb2xvcjogI2ZmZjtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAuc2Vjb25kYXJ5LWNvbG9yLXBhc3RlbCB7XG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjQzhFNkVEO1xuICAgICAgcCB7XG4gICAgICAgIGNvbG9yOiAjOERCMkJBO1xuICAgICAgfVxuICAgIH1cblxuICAgIC53aGl0ZSB7XG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjZmZmO1xuICAgICAgYm9yZGVyOjFweCBzb2xpZCAjQzhFNkVEO1xuICAgICAgcCB7XG4gICAgICAgIGNvbG9yOiAjOERCMkJBO1xuICAgICAgfVxuICAgIH1cblxuICAgIC5zZW5zb3JzID4gZGl2IHtcbiAgICAgIHdpZHRoOiA2MHB4O1xuICAgICAgaGVpZ2h0OiA2MHB4O1xuICAgICAgYm9yZGVyLXJhZGl1czogMzBweDtcbiAgICAgIGZsb2F0OiBsZWZ0O1xuICAgICAgbWFyZ2luOiAxMHB4O1xuICAgIH1cblxuICAgIC5zZW5zb3JzIGRpdjpudGgtY2hpbGQoMSkge1xuICAgICBiYWNrZ3JvdW5kOiAjZmZjMTA3O1xuICAgIH1cblxuICAgIC5zZW5zb3JzIGRpdjpudGgtY2hpbGQoMikge1xuICAgICBiYWNrZ3JvdW5kOiAjNGZjM2Y3O1xuICAgIH1cblxuICAgIC5zZW5zb3JzIGRpdjpudGgtY2hpbGQoMykge1xuICAgICBiYWNrZ3JvdW5kOiAjZmZlZTU4O1xuICAgIH1cblxuICAgIC5zZW5zb3JzIGRpdjpudGgtY2hpbGQoNCkge1xuICAgICBiYWNrZ3JvdW5kOiAjZjA2MjkyO1xuICAgIH1cblxuICAgIC5zZW5zb3JzIGRpdjpudGgtY2hpbGQoNSkge1xuICAgICBiYWNrZ3JvdW5kOiAjNGNhZjUwO1xuICAgIH1cblxuICAgIC5zZW5zb3JzIGRpdjpudGgtY2hpbGQoNikge1xuICAgICBiYWNrZ3JvdW5kOiAjOGJjMzRhO1xuICAgIH1cblxuICAgIC5zZW5zb3JzIGRpdjpudGgtY2hpbGQoNykge1xuICAgICBiYWNrZ3JvdW5kOiAjOTU3NWNkO1xuICAgIH1cblxuICAgIC5zZW5zb3JzIGRpdjpudGgtY2hpbGQoOCkge1xuICAgICBiYWNrZ3JvdW5kOiAjZmZmOWM0O1xuICAgIH1cblxuICAgIC5zZW5zb3JzIGRpdjpudGgtY2hpbGQoOSkge1xuICAgICBiYWNrZ3JvdW5kOiAjZmZlZTU4O1xuICAgIH1cblxuICAgIC5zZW5zb3JzIGRpdjpudGgtY2hpbGQoMTApIHtcbiAgICAgYmFja2dyb3VuZDogI2ZmNTcyMjtcbiAgICB9XG4gIH1cblxufVxuXG5cbi8qIEN1c3RvbSBBbmltYXRpb25zICovXG5cbi5sb2FkaW5nLWRvdHMgc3BhbjphZnRlciB7XG4gIGNvbnRlbnQ6ICcuJztcbiAgYW5pbWF0aW9uOiBkb3RzIDJzIHN0ZXBzKDUsIGVuZCkgaW5maW5pdGU7fVxuXG5Aa2V5ZnJhbWVzIGRvdHMge1xuICAwJSwgMjAlIHtcbiAgICBjb2xvcjogcmdiYSgwLDAsMCwwKTtcbiAgICB0ZXh0LXNoYWRvdzpcbiAgICAgIC4yNWVtIDAgMCByZ2JhKDAsMCwwLDApLFxuICAgICAgLjVlbSAwIDAgcmdiYSgwLDAsMCwwKTt9XG4gIDQwJSB7XG4gICAgY29sb3I6IHdoaXRlO1xuICAgIHRleHQtc2hhZG93OlxuICAgICAgLjI1ZW0gMCAwIHJnYmEoMCwwLDAsMCksXG4gICAgICAuNWVtIDAgMCByZ2JhKDAsMCwwLDApO31cbiAgNjAlIHtcbiAgICB0ZXh0LXNoYWRvdzpcbiAgICAgIC4yNWVtIDAgMCAkeWVsbG93LFxuICAgICAgLjVlbSAwIDAgcmdiYSgwLDAsMCwwKTt9XG4gIDgwJSwgMTAwJSB7XG4gICAgdGV4dC1zaGFkb3c6XG4gICAgICAuMjVlbSAwIDAgJHllbGxvdyxcbiAgICAgIC41ZW0gMCAwICRibHVlO319XG5cblxuLyogRG8gbm90IHJlbW92ZSB0aGlzIGNvbW1lbnRzIGJlbGxvdy4gSXQncyB0aGUgbWFya2VycyB1c2VkIGJ5IGd1bHAtaW5qZWN0IHRvIGluamVjdFxuICAgYWxsIHlvdXIgc2FzcyBmaWxlcyBhdXRvbWF0aWNhbGx5ICovXG4vLyBpbmplY3RvclxuLy8gZW5kaW5qZWN0b3JcbiIsIi8vIEdlbmVyYWwgaGVscGVyc1xuLmJvcmRlci1ibGFja3tcbiAgYm9yZGVyOiAzcHggc29saWQgJG9mZl9ibGFjaztcbn1cbi5ib3JkZXItd2hpdGV7XG4gIGJvcmRlcjogM3B4IHNvbGlkIHdoaXRlO1xufVxuLmNvbG9yLXdoaXRle1xuICBjb2xvcjogd2hpdGUgIWltcG9ydGFudDtcbn1cbi5jb2xvci1kcm9wZG93bntcbiAgY29sb3I6ICRncmV5X2Rhcmtlc3QgIWltcG9ydGFudDtcbn1cbi5jb2xvci1ibGFja3tcbiAgY29sb3I6ICRvZmZfYmxhY2s7XG59XG4uY29sb3ItYmx1ZXtcbiAgY29sb3I6JHRlcmNpYXJ5X2NvbG9yO1xufVxuLmNvbG9yLWN5YW57XG4gIGNvbG9yOiAkdGVyY2lhcnlfY29sb3I7XG59XG4uY29sb3ItZ3JlZW4ge1xuICBjb2xvcjogJGdyZWVuO1xuICBzdmcge1xuICAgIGZpbGw6ICRncmVlbjtcbiAgfVxufVxuLmNvbG9yLXJlZCB7XG4gIGNvbG9yOiAkcmVkO1xuICBzdmcge1xuICAgIGZpbGw6ICRyZWQgIWltcG9ydGFudDtcbiAgfVxufVxuLmZsb2F0LWxlZnR7XG4gIGZsb2F0OiBsZWZ0O1xufVxuLmZsb2F0LXJpZ2h0e1xuICBmbG9hdDogcmlnaHQ7XG59XG4uZC1mbGV4e1xuICBkaXNwbGF5OmZsZXg7XG59XG4uZnVsbC13aWR0aHtcbiAgd2lkdGg6IDEwMCU7XG59XG4ubWF4LXdpZHRoLTUwMHB4IHtcbiAgbWF4LXdpZHRoOiA1MDBweDtcbn1cbi5taW4taGVpZ2h0LTgwIHtcbiAgbWluLWhlaWdodDogODAlO1xufVxuXG4vLyBGb250ICYgdGV4dFxuLmZvbnQtcm9ib3RvLWNvbmRlbnNlZHtcbiAgZm9udC1mYW1pbHk6J1JvYm90byBDb25kZW5zZWQnO1xufVxuLmZvbnQta2FuaXR7XG4gIGZvbnQtZmFtaWx5OidLYW5pdCc7XG59XG5cbi50ZXh0LWNlbnRlcntcbiAgdGV4dC1hbGlnbjogY2VudGVyO1xufVxuLnRleHQtbGVmdHtcbiAgdGV4dC1hbGlnbjogbGVmdDtcbn1cblxuXG5cbi8vIEJhY2tncm91bmQgaGVscGVyc1xuLmJnLXdoaXRle1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAjZmZmO1xufVxuLmJnLWJsYWNre1xuICBiYWNrZ3JvdW5kLWNvbG9yOiBibGFjaztcbn1cbi5iZy1ncmVlbntcbiAgYmFja2dyb3VuZC1jb2xvcjogJGdyZWVuO1xufVxuLmJnLWJsdWV7XG4gIGJhY2tncm91bmQtY29sb3I6ICRibHVlO1xufVxuLmJnLXJlZHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJHJlZCAhaW1wb3J0YW50O1xufVxuLmJnLXJlZC1saWdodHtcbiAgYmFja2dyb3VuZC1jb2xvcjogcmdiYSgkcmVkLC4yKTtcbn1cbi5iZy15ZWxsb3d7XG4gIGJhY2tncm91bmQtY29sb3I6ICR5ZWxsb3c7XG59XG4uYmctZ3JleSB7XG4gIGJhY2tncm91bmQtY29sb3I6ICRncmV5ICFpbXBvcnRhbnQ7XG59XG4uYmctZ3JleS1saWdodGVzdCB7XG4gIGJhY2tncm91bmQtY29sb3I6ICRncmV5X2xpZ2h0ZXN0ICFpbXBvcnRhbnQ7XG59XG5cbi5vdXRsaW5le1xuICBvdXRsaW5lOiAxcHggc29saWQgcmVkO1xufVxuXG4vLyBCdXR0b24gaGVscGVyc1xuLmJ0bi1vdXRsaW5lLWJsdWV7XG4gIGJvcmRlcjoycHggc29saWQgJGJsdWU7XG4gIGNvbG9yOiAkYmx1ZTtcbiAgcGFkZGluZzoxMHB4IDI5cHg7XG4gIHN2ZyB7XG4gICAgZmlsbDogJGJsdWVcbiAgfVxufVxuLmJ0bi1vdXRsaW5lLWJsdWU6aG92ZXJ7XG4gIGJhY2tncm91bmQ6ICRibHVlICFpbXBvcnRhbnQ7XG4gIGNvbG9yOiB3aGl0ZTtcbiAgc3ZnIHtcbiAgICBmaWxsOiB3aGl0ZVxuICB9XG59XG4uYnRuLW91dGxpbmUtd2hpdGV7XG4gIGJvcmRlcjoycHggc29saWQgd2hpdGU7XG4gIGNvbG9yOiB3aGl0ZTtcbiAgcGFkZGluZzoxMHB4IDI5cHg7XG59XG4uYnRuLW91dGxpbmUtd2hpdGU6aG92ZXJ7XG4gIGJhY2tncm91bmQ6IHdoaXRlICFpbXBvcnRhbnQ7XG4gIGNvbG9yOiAkb2ZmX2JsYWNrO1xufVxuLmJ0bi1vdXRsaW5lLXdoaXRlLWJsdWV7XG4gIGJvcmRlcjoycHggc29saWQgd2hpdGU7XG4gIGNvbG9yOiB3aGl0ZTtcbiAgcGFkZGluZzoxMHB4IDI5cHg7XG59XG4uYnRuLW91dGxpbmUtd2hpdGUtYmx1ZTpob3ZlcntcbiAgYmFja2dyb3VuZDogJGJsdWUgIWltcG9ydGFudDtcbn1cbi5idG4tb3V0bGluZS15ZWxsb3d7XG4gIGJvcmRlcjoycHggc29saWQgJHllbGxvdztcbiAgY29sb3I6ICR5ZWxsb3c7XG4gIHBhZGRpbmc6MTBweCAyOXB4O1xufVxuLmJ0bi1vdXRsaW5lLXllbGxvdzpob3ZlcntcbiAgYmFja2dyb3VuZDogJHllbGxvdyAhaW1wb3J0YW50O1xuICBjb2xvcjogJG9mZl9ibGFjaztcbn1cbi5idG4teWVsbG93e1xuICBib3JkZXI6MnB4IHNvbGlkICR5ZWxsb3c7XG4gIGJhY2tncm91bmQ6ICR5ZWxsb3c7XG4gIGNvbG9yOiAkb2ZmX2JsYWNrO1xuICBwYWRkaW5nOjEycHggMjlweDtcbn1cblxuLmJ0bi15ZWxsb3c6aG92ZXJ7XG4gIGJhY2tncm91bmQ6ICRvZmZfYmxhY2sgIWltcG9ydGFudDtcbiAgY29sb3I6ICR5ZWxsb3c7XG59XG4uYnRuLWJsYWNrLW91dGxpbmV7XG4gIGJvcmRlcjoycHggc29saWQgJG9mZl9ibGFjaztcbiAgY29sb3I6ICRvZmZfYmxhY2s7XG4gIHBhZGRpbmc6MTBweCAyOXB4O1xufVxuLmJ0bi1ibGFjay1vdXRsaW5lOmhvdmVye1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkb2ZmX2JsYWNrICFpbXBvcnRhbnQ7XG4gIGNvbG9yOiB3aGl0ZTtcbn1cbi5idG4tYmx1ZXtcbiAgYm9yZGVyOjJweCBzb2xpZCAkYmx1ZTtcbiAgYmFja2dyb3VuZDogJGJsdWU7XG4gIGNvbG9yOiB3aGl0ZTtcbiAgcGFkZGluZzoxMnB4IDI5cHg7XG59XG5cbi5idG4tYmx1ZTpob3ZlcntcbiAgY29sb3I6ICRibHVlO1xuICBiYWNrZ3JvdW5kOiAkZ3JleV9saWdodGVyICFpbXBvcnRhbnQ7XG59XG4uYnRuLXdoaXRlLWJsdWV7XG4gIGJvcmRlcjoycHggc29saWQgd2hpdGU7XG4gIGJhY2tncm91bmQ6IG5vbmU7XG4gIGNvbG9yOiB3aGl0ZTtcbiAgcGFkZGluZzoxMnB4IDI5cHg7XG59XG5cbi5idG4td2hpdGUtYmx1ZTpob3ZlcntcbiAgY29sb3I6ICRibHVlO1xuICBiYWNrZ3JvdW5kOiB3aGl0ZSAhaW1wb3J0YW50O1xuICBib3JkZXI6MnB4IHNvbGlkICRibHVlO1xufVxuXG4vLyBUT0RPOiBDb3NtZXRpYyBGaXggYnV0dG9ucyBpbiBnZW5lcmFsXG4uYnRuLXJvdW5kLW5ldyB7XG4gIGJvcmRlci1yYWRpdXM6IDQwcHg7XG4gIGZvbnQtZmFtaWx5OiAnUm9ib3RvIENvbmRlbnNlZCc7XG4gIGZvbnQtc2l6ZTogMTZweDtcbiAgLy8gaGVpZ2h0OiA0NHB4O1xuICBsaW5lLWhlaWdodDogNDBweDtcbiAgZm9udC13ZWlnaHQ6ICRmb250LXdlaWdodC1oZWF2eTtcbiAgbGV0dGVyLXNwYWNpbmc6IDAuMDMyNWVtO1xuICBwYWRkaW5nOjBweCAzNXB4O1xuICB0cmFuc2l0aW9uOiBib3gtc2hhZG93IDAuNHMgY3ViaWMtYmV6aWVyKDAuMjUsIDAuOCwgMC4yNSwgMSksIGJhY2tncm91bmQtY29sb3IgMC40cyBjdWJpYy1iZXppZXIoMC4yNSwgMC44LCAwLjI1LCAxKTtcbn1cblxuLmJ0bi1yb3VuZHtcbiAgYm9yZGVyOiAwO1xuICBib3JkZXItcmFkaXVzOiAyNXB4O1xuICBwYWRkaW5nOjVweCAzNXB4O1xuICBmb250LXNpemU6MThweDtcbn1cblxuLmJ0bi1jeWFue1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAkdGVyY2lhcnlfY29sb3I7XG4gIGNvbG9yOndoaXRlO1xuICBib3JkZXI6MnB4IHNvbGlkICR0ZXJjaWFyeV9jb2xvcjtcbn1cblxuLmJ0bi1jeWFuOmhvdmVye1xuICBjb2xvcjogJHRlcmNpYXJ5X2NvbG9yO1xuICBiYWNrZ3JvdW5kOiB3aGl0ZSAhaW1wb3J0YW50O1xuICBib3JkZXI6MnB4IHNvbGlkICR0ZXJjaWFyeV9jb2xvcjtcbn1cblxuLy8gZGlzcGxheSBoZWxwZXJzXG4uYnRuLWZ1bGwge1xuICBkaXNwbGF5OiBibG9jaztcbiAgd2lkdGg6IDEwMCU7XG4gIGJvcmRlci1yYWRpdXM6IDBweDtcbiAgbWFyZ2luOiAwcHg7XG4gIHBhZGRpbmc6MTJweCAyOXB4O1xufVxuXG4uYnRuLXNtYWxsIHtcbiAgbWluLXdpZHRoOiB1bnNldCAhaW1wb3J0YW50O1xuICBtaW4taGVpZ2h0OiB1bnNldCAhaW1wb3J0YW50O1xuICAvL291dGxpbmU6IDFweCBzb2xpZCByZWQ7XG59XG5cbi8vIE1hcmdpbiAtIHBhZGRpbmcgaGVscGVyc1xuLm0tMHtcbiAgbWFyZ2luOiAwO1xufVxuLm0tMTB7XG4gIG1hcmdpbjogMTBweDtcbn1cbi5tbC0wIHtcbiAgbWFyZ2luLWxlZnQ6IDBweDtcbn1cbi5tbC0xNSB7XG4gIG1hcmdpbi1sZWZ0OiAxNXB4O1xufVxuLm10LTEweyBtYXJnaW4tdG9wOiAxMHB4OyB9XG4ubXQtMjB7IG1hcmdpbi10b3A6IDIwcHg7IH1cbi5tdC0zMHsgbWFyZ2luLXRvcDogMzBweDsgfVxuLm10LTUweyBtYXJnaW4tdG9wOiA1MHB4OyB9XG4ubXItMTB7IG1hcmdpbi1yaWdodDogMTBweDsgfVxuLm1yLTIweyBtYXJnaW4tcmlnaHQ6IDIwcHg7IH1cbi5tci0zMHsgbWFyZ2luLXJpZ2h0OiAzMHB4OyB9XG4ubXItNTB7IG1hcmdpbi1yaWdodDogNTBweDsgfVxuLm1iLTEweyBtYXJnaW4tYm90dG9tOiAxMHB4OyB9XG4ubWItMjB7IG1hcmdpbi1ib3R0b206IDIwcHg7IH1cbi5tYi0zMHsgbWFyZ2luLWJvdHRvbTogMzBweDsgfVxuLm1iLTUweyBtYXJnaW4tYm90dG9tOiA1MHB4OyB9XG4ubWwtMTAgeyBtYXJnaW4tbGVmdDogMTBweDsgfVxuLm1sLTIwIHsgbWFyZ2luLWxlZnQ6IDIwcHg7IH1cbi5tbC0zMCB7IG1hcmdpbi1sZWZ0OiAzMHB4OyB9XG4ubWwtNTB7IG1hcmdpbi1sZWZ0OiA1MHB4OyB9XG4ubXktMjB7XG4gIG1hcmdpbi10b3A6IDIwcHggIWltcG9ydGFudDtcbiAgbWFyZ2luLWJvdHRvbTogMjBweCAhaW1wb3J0YW50O1xufVxuLm13LTEwMHtcbiAgbWF4LXdpZHRoOiAxMDAlO1xufVxuLnAtMHsgcGFkZGluZzogMDsgfVxuLnAtMTB7IHBhZGRpbmc6IDEwcHg7IH1cbi5wLTIweyBwYWRkaW5nOiAyMHB4OyB9XG4ucC0zMHsgcGFkZGluZzogMzBweDsgfVxuLnAtNDB7IHBhZGRpbmc6IDQwcHg7IH1cbi5wLTUweyBwYWRkaW5nOiA1MHB4OyB9XG4ucC02MHsgcGFkZGluZzogNjBweDsgfVxuLnBsLTIweyBwYWRkaW5nLWxlZnQ6IDIwcHg7IH1cbi5wdC04MHsgcGFkZGluZy10b3A6ODBweDsgfVxuLnB5LTQwe1xuICBwYWRkaW5nLXRvcDo0MHB4O1xuICBwYWRkaW5nLWJvdHRvbTo0MHB4O1xufVxuLnB4LTIwe1xuICBwYWRkaW5nLWxlZnQ6MjBweDtcbiAgcGFkZGluZy1yaWdodDoyMHB4O1xufVxuLnB4LTQwe1xuICBwYWRkaW5nLWxlZnQ6NDBweDtcbiAgcGFkZGluZy1yaWdodDo0MHB4O1xufVxuXG5cbi8vIGltZyBoZWxwZXJzXG5cbi5pbWctY2lyY2xlIHtcbiAgYm9yZGVyLXJhZGl1czogNTAlO1xufVxuXG5mb290ZXJ7XG4gIHB7XG4gICAgZm9udC1zaXplOjEycHggIWltcG9ydGFudDtcbiAgICBsaW5lLWhlaWdodDoxMnB4ICFpbXBvcnRhbnQ7XG4gIH1cbn1cblxuLy8gcG9zdGlvbiBoZWxwZXJzXG5cbi5hYnNvbHV0ZSB7XG4gIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbn1cblxuLnJlbGF0aXZlIHtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xufVxuXG4vLyBzaXplIGhlbHBlcnNcbi5zLTQ4IHtcbiAgd2lkdGg6IDQ4cHg7XG4gIGhlaWdodDogNDhweDtcbn1cbiIsIiR5ZWxsb3c6ICNGRkMxMDA7XG4keWVsbG93X2xpZ2h0OiAjRkZENDREO1xuJHllbGxvd19saWdodGVyOiAjRkZFQ0IzO1xuJGJsdWU6ICMwMDE5RkY7XG4kYmx1ZV9saWdodDogIzU1QzRGNTtcblxuJHJlZDogI0ZGM0Q0QztcbiRncmVlbjogIzAwRTU5Nztcbi8vICRvZmZfYmxhY2s6ICMyRTJFMkU7XG4vLyAkb2ZmX2Rhcmtlcl9ibGFjazogIzFDMUMxQztcbiRvZmZfbGlnaHRlcl9ibGFjazogIzJFMkUyRTtcbiRvZmZfYmxhY2s6ICMxQzFDMUM7XG5cbiRncmV5X2xpZ2h0ZXN0OiAjRjlGOUZCO1xuJGdyZXlfbGlnaHRlcjogI0VDRUNFRTtcbiRncmV5OiAjRDJEM0Q1O1xuJGdyZXlfZGFya2VyOiAjNkU2RTZFO1xuJGdyZXlfZGFya2VzdDogIzdFN0U3RTtcblxuJHNlY29uZGFyeV9jb2xvcjogJGdyZXlfZGFya2VyO1xuJHRlcmNpYXJ5X2NvbG9yOiAkYmx1ZTtcblxuJHNlY29uZGFyeV9jb2xvcl9saWdodDogJGdyZXlfZGFya2VyO1xuJHRlcmNpYXJ5X2NvbG9yX2xpZ2h0OiAkZ3JleTtcblxuJGluZm9fcHJpbWFyeTogJGdyZXk7XG4kaW5mb19zZWNvbmRhcnk6ICRzZWNvbmRhcnlfY29sb3JfbGlnaHQ7XG5cbiRhbGVydF9yZWQ6ICRyZWQ7XG4kYWxlcnRfZ3JlZW46ICM0REJENEU7XG5cbiRmb250X2NvbG9yX3doaXRlOiB3aGl0ZTtcbiRmb250X2NvbG9yX2JsdWVfZ3JleTogJGdyZXk7XG4kZm9udF9jb2xvcl9ibHVlX2Rhcms6ICRncmV5X2RhcmtlcjtcbiRmb250X2NvbG9yX2JsdWVfbGlnaHQ6ICRibHVlO1xuXG4kYmFja2dyb3VuZF9ibHVlX2RhcmtfbWVudTogJG9mZl9ibGFjaztcbiRiYWNrZ3JvdW5kX2JsdWVfbGlnaHRlcl9tZW51OiAkb2ZmX2xpZ2h0ZXJfYmxhY2s7XG4kYmFja2dyb3VuZF9ibHVlX2Rhcmtfb3ZlcnZpZXc6ICRvZmZfYmxhY2s7XG4kYmFja2dyb3VuZF9ibHVlX2RhcmtfYmFja2dyb3VuZDogJG9mZl9ibGFjaztcbiRiYWNrZ3JvdW5kX2JsdWVfZGFya19uYXZiYXI6ICRvZmZfYmxhY2s7XG4kYmFja2dyb3VuZF9ibHVlX2RhcmtfY2hhcnQ6ICRvZmZfYmxhY2s7XG4kYmFja2dyb3VuZF9ibHVlX2xpZ2h0OiAkZ3JleV9saWdodGVyO1xuJGJhY2tncm91bmRfYmx1ZV9taWR0b25lOiAkZ3JleTtcblxuJGxhcmdlX3ByaW9yaXR5OiAxMDtcbiRtb3JlX3ByaW9yaXR5OiA1O1xuJG1lZGl1bV9wcmlvcml0eTogMjtcbiRzbWFsbF9wcmlvcml0eTogLTE7XG5cbiRkcm9wZG93bl9ibHVlOiAjM2Y1MWI1O1xuXG4kZm9udC13ZWlnaHQtaGVhdnk6IDYwMDtcbiR0aW1lbGluZS1oZWlnaHQ6IDEyMHB4O1xuJGNvbnRlbnQtbWFyZ2luLXRvcDogMjIwcHg7XG4iLCIvLyBOZXcgbGFuZGluZyBwYWdlXG5cbmh0bWx7XG4gIHNjcm9sbC1iZWhhdmlvcjpzbW9vdGhcbn1cbmJvZHkuYWN0aXZlIHtcbiAgb3ZlcmZsb3cteTogaGlkZGVuO1xuICBvdmVyZmxvdy14OiBoaWRkZW47XG4gIHBvc2l0aW9uOiBmaXhlZDtcbn1cbi5ncmV5LXdhdmVze1xuICBiYWNrZ3JvdW5kOiBncmV5IHVybChcIi9hc3NldHMvaW1hZ2VzL3Nja19iZy5wbmdcIikgO1xuICBiYWNrZ3JvdW5kLXJlcGVhdDogcmVwZWF0O1xuICBiYWNrZ3JvdW5kLXNpemU6IDY1cHg7XG59XG4uYmFyIHtcbiAgZGlzcGxheTogYmxvY2s7XG4gIHdpZHRoOiAyNXB4O1xuICBoZWlnaHQ6IDNweDtcbiAgbWFyZ2luOiA1cHggYXV0bztcbiAgLXdlYmtpdC10cmFuc2l0aW9uOiBhbGwgMC41cyBlYXNlLWluLW91dDtcbiAgdHJhbnNpdGlvbjogYWxsIDAuNXMgZWFzZS1pbi1vdXQ7XG4gIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xufVxuXG4uaGFtYnVyZ2VyIHtcbiAgZGlzcGxheTogbm9uZTtcbn1cblxuLmxvZ28tYm94IHtcbiAgYmFja2dyb3VuZC1jb2xvcjogJG9mZl9ibGFjaztcbiAgIHdpZHRoOiAxMDAlO1xuICAgZGlzcGxheTogZmxleDtcbiAgIGp1c3RpZnktY29udGVudDogY2VudGVyO1xuICAgYWxpZ24tY29udGVudDogY2VudGVyO1xufVxuLmxvZ28tYm94IGgxIHtcbiAgY29sb3I6IHdoaXRlICFpbXBvcnRhbnQ7XG4gIGZvbnQtd2VpZ2h0OiA4MDAgIWltcG9ydGFudDtcbiAgZm9udC1zaXplOiA1MHB4O1xuICBwYWRkaW5nOiAxNXB4O1xuICB0ZXh0LWFsaWduOiBsZWZ0O1xufVxuLmNhcmQtdGV4dCB7XG4gIHBhZGRpbmc6IDMwcHggNTBweDtcbiAgYWxpZ24tc2VsZjogY2VudGVyO1xufVxuXG4ubmF2aWdhdGlvbiB7XG4gIGRpc3BsYXk6IGZsZXg7XG4gIGZsZXgtZGlyZWN0aW9uOiByb3c7XG4gIGp1c3RpZnktY29udGVudDogc3BhY2UtYmV0d2VlbjtcbiAgd2lkdGg6IGF1dG87XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIHBhZGRpbmc6IDZweCAwJSA2cHggMTAlO1xufVxuLmRpc3BsYXktbm9uZXtcbiAgZGlzcGxheTogbm9uZSAhaW1wb3J0YW50O1xufVxuLmRpc3BsYXktYmxvY2t7XG4gIGRpc3BsYXk6IGJsb2NrICFpbXBvcnRhbnQ7XG59XG5cbi5uZXctbGFuZGluZy1wYWdlIHtcbiAgZGlzcGxheTogZmxleDtcbiAgb3ZlcmZsb3c6IGhpZGRlbjtcbiAgZmxleC1kaXJlY3Rpb246IGNvbHVtbjtcbiAgcGFkZGluZy1ib3R0b206IDYwcHg7XG4gIGgxLGgyLGgzLGg0LGg1LGg2e1xuICAgIGxldHRlci1zcGFjaW5nOiAwLjAzMjVlbTtcbiAgICBmb250LWZhbWlseTogJ0thbml0JztcbiAgICBmb250LXdlaWdodDogOTAwO1xuICAgIGNvbG9yOiAkb2ZmX2JsYWNrO1xuICB9XG4gIGgxe1xuICAgIC8vIGZvbnQtc2l6ZTogNDVweDtcbiAgICBsaW5lLWhlaWdodDogNTVweDtcbiAgICBtYXJnaW46MHB4IDBweDtcbiAgfVxuICBoMntcbiAgICBmb250LXNpemU6IDQzcHg7XG4gICAgbGluZS1oZWlnaHQ6IDQwcHg7XG4gICAgbWFyZ2luOjBweCAwcHg7XG4gIH1cbiAgaDN7XG4gICAgZm9udC1zaXplOiAzMHB4O1xuICAgIGxpbmUtaGVpZ2h0OiAzMHB4O1xuICAgIG1hcmdpbi10b3A6MTBweDtcbiAgICBtYXJnaW4tYm90dG9tOiAxMHB4O1xuICB9XG4gIHB7XG4gICAgY29sb3I6ICRvZmZfYmxhY2s7XG4gICAgZm9udC1mYW1pbHk6ICdSb2JvdG8gQ29uZGVuc2VkJztcbiAgICBmb250LXNpemU6IDIwcHg7XG4gICAgbGluZS1oZWlnaHQ6IDI0cHg7XG4gIH1cbiAgLnAtaGVhZGVyIHtcbiAgICBmb250LXNpemU6IDI0cHg7XG4gIH1cbiAgYXtcbiAgICB0ZXh0LWRlY29yYXRpb246IG5vbmUgIWltcG9ydGFudDtcbiAgfVxuICAubGFuZGluZy1tZW51LWJ0bntcbiAgICBib3JkZXItcmFkaXVzOiA0MHB4O1xuICAgIGZvbnQtZmFtaWx5OiAnUm9ib3RvIENvbmRlbnNlZCc7XG4gICAgZm9udC1zaXplOiAxNnB4O1xuICAgIC8vIGhlaWdodDogNDRweDtcbiAgICBsaW5lLWhlaWdodDogNDBweDtcbiAgICBmb250LXdlaWdodDogJGZvbnQtd2VpZ2h0LWhlYXZ5O1xuICAgIGxldHRlci1zcGFjaW5nOiAwLjAzMjVlbTtcbiAgICBwYWRkaW5nOjBweCAzMHB4O1xuICAgIGNvbG9yOiB3aGl0ZTtcbiAgICB0cmFuc2l0aW9uOiBib3gtc2hhZG93IDAuNHMgY3ViaWMtYmV6aWVyKDAuMjUsIDAuOCwgMC4yNSwgMSksIGJhY2tncm91bmQtY29sb3IgMC40cyBjdWJpYy1iZXppZXIoMC4yNSwgMC44LCAwLjI1LCAxKTtcbiAgfVxuICAuYnRuLWp1c3RpZnktbGVmdHtcbiAgICBtYXJnaW4tbGVmdDogMHB4O1xuICB9XG4gIC5sYW5kaW5nLW1lbnUgLmJ0bi1yb3VuZC1uZXcubWQtYnV0dG9uLmJ0bi1raXQge1xuICAgIG1hcmdpbi1sZWZ0OiAzOHB4O1xuICB9XG4gIC5sYW5kaW5nLW1lbnUtYnRuOmhvdmVye1xuICAgIGJhY2tncm91bmQ6IG5vbmU7XG4gICAgY29sb3I6ICNGRkMxMDA7XG4gIH1cbiAgLnNjLWxvZ297XG4gICAgaGVpZ2h0OiAzN3B4O1xuICAgIHRvcDogNXB4O1xuICB9XG5cbiAgLnNjLW9mZi1jdGEtcGxhdGZvcm0ge1xuICAgIHBvc2l0aW9uOiBmaXhlZDtcbiAgICB0b3A6IDA7XG4gICAgei1pbmRleDogMztcbiAgICBkaXNwbGF5OiBmbGV4O1xuICAgIGZsZXgtZGlyZWN0aW9uOiByb3c7XG4gICAgZmxleC13cmFwOm5vd3JhcDtcbiAgICB3aWR0aDogY2FsYygxMDAlIC0gMjBweCk7XG4gICAganVzdGlmeS1jb250ZW50OiBzcGFjZS1iZXR3ZWVuO1xuICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogJG9mZl9ibGFjaztcbiAgICBtYXJnaW46IDEwcHggMTBweDtcblxuICB9XG4gICNnZXQteW91ci1raXQge1xuICAgIHNjcm9sbC1tYXJnaW4tdG9wOiA3OHB4O1xuICB9XG4gIC5saWdodGVye1xuICAgIG1heC13aWR0aDogMTAlO1xuICAgIHRyYW5zaXRpb246IGFsbCAwLjNzIGVhc2UtaW4tb3V0O1xuICAgIGZsZXgtZGlyZWN0aW9uOiBjb2x1bW47XG4gICAgcGFkZGluZzogNXB4O1xuICAgIG9wYWNpdHk6IDkwJTsgXG4gIH1cbiAgLmxpZ2h0ZXI6aG92ZXIge1xuICAgIG9wYWNpdHk6IDEwMCU7XG4gIH1cblxuICAubGFuZGluZy1tZW51IHtcbiAgICBkaXNwbGF5OiBmbGV4O1xuICAgIGZsZXgtZGlyZWN0aW9uOiByb3c7XG4gICAgZmxleC13cmFwOiB3cmFwO1xuICAgIHdpZHRoOiBmaXQtY29udGVudDtcbiAgICBhbGlnbi1zZWxmOiBjZW50ZXI7XG4gICAganVzdGlmeS1jb250ZW50OiBzcGFjZS1iZXR3ZWVuO1xuICAgIHBhZGRpbmc6IDZweCAxMCUgNnB4IDAlO1xuICB9XG4gIC8vIEltYWdlcyBvbiBsYW5kaW5nIHBhZ2VcbiAgLmltZy1uZXdfc2Nre1xuICAgIG1pbi1oZWlnaHQ6IDMwdnc7XG4gICAgYmFja2dyb3VuZDogYmx1ZSB1cmwoXCIvYXNzZXRzL2ltYWdlcy9sYW5kaW5nL0xlZ2FjeS1TQ0suZ2lmXCIpIGNlbnRlci9jb3ZlcjtcbiAgfVxuICAuaW1nLXNja19lZHV7XG4gICAgbWluLWhlaWdodDogMzB2dztcbiAgICBiYWNrZ3JvdW5kOiBibHVlIHVybChcIi9hc3NldHMvaW1hZ2VzL2xhbmRpbmcvc21hcnQtY2l0aXplbi1kYXRhLmpwZ1wiKSBjZW50ZXIvY292ZXI7XG4gIH1cbiAgLmltZy1zY2tfY29te1xuICAgIGJhY2tncm91bmQ6IGJsdWUgdXJsKFwiL2Fzc2V0cy9pbWFnZXMvbGFuZGluZy9sb2NhbC1jb21tdW5pdGVzLmpwZ1wiKSBjZW50ZXIvY292ZXI7XG4gIH1cbiAgLmltZy1yZXNlYXJjaHtcbiAgICBiYWNrZ3JvdW5kOiBibHVlIHVybChcIi9hc3NldHMvaW1hZ2VzL2xhbmRpbmcvcmVzZWFyY2gtaW5zdGl0dXRpb25zLmpwZ1wiKSBjZW50ZXIvY292ZXI7XG4gIH1cbiAgLmltZy1nb3Zlcm5te1xuICAgIGJhY2tncm91bmQ6IGJsdWUgdXJsKFwiL2Fzc2V0cy9pbWFnZXMvbGFuZGluZy9lZHVjYXRvcnMuanBnXCIpIGNlbnRlci9jb3ZlcjtcbiAgfVxuICAuaW1nLXBsYXRmb3Jte1xuICAgIGJhY2tncm91bmQ6IGJsdWUgdXJsKFwiL2Fzc2V0cy9pbWFnZXMvbGFuZGluZy9TQ0stUGxhdGZvcm0tbW9iaWxlLmpwZ1wiKSBjZW50ZXIvY292ZXI7XG4gIH1cbiAgLmltZy1kb2Nze1xuICAgIGJhY2tncm91bmQ6IGJsdWUgdXJsKFwiL2Fzc2V0cy9pbWFnZXMvbGFuZGluZy9zbWFydGNpdGl6ZW4tZG9jcy5qcGdcIikgdG9wL2NvdmVyO1xuICB9XG4gIC5pbWcta2l0cy1jb21wbGV0ZXtcbiAgICBiYWNrZ3JvdW5kOiBibHVlIHVybChcIi9hc3NldHMvaW1hZ2VzL2xhbmRpbmcvc21hcnQtY2l0aXplbi0yLTIta2l0LmdpZlwiKSBjZW50ZXIvY292ZXI7XG4gIH1cbiAgLmltZy1raXRzLXJlc2VhcmNoe1xuICAgIGJhY2tncm91bmQ6IGJsdWUgdXJsKFwiL2Fzc2V0cy9pbWFnZXMvbGFuZGluZy9TQ0stY3VzdG9taXplZC1wcm9qZWN0cy5qcGdcIikgY2VudGVyL2NvdmVyO1xuICB9XG4gIC8vIFRpbGVzIG9uIGxhbmRpbmcgcGFnZVxuICAudGlsZXtcbiAgICBwYWRkaW5nOiA1MHB4IDUwcHg7XG4gICAgYm9yZGVyLWJvdHRvbTogM3B4IHNvbGlkICRvZmZfYmxhY2s7XG4gICAgYm9yZGVyLXJpZ2h0OiAzcHggc29saWQgJG9mZl9ibGFjaztcbiAgfVxuICAudGlsZS1pbWFnZS50aWxle1xuICAgIG1pbi1oZWlnaHQ6MzAwcHg7XG4gICAgcGFkZGluZzowICFpbXBvcnRhbnQ7XG4gIH1cblxuICAudGlsZS1sZWZ0e1xuICAgIGJvcmRlci1sZWZ0OiAzcHggc29saWQgJG9mZl9ibGFjaztcbiAgfVxuICAudGlsZS10b3B7XG4gICAgYm9yZGVyLXRvcDogM3B4IHNvbGlkICRvZmZfYmxhY2s7XG4gIH1cblxuICAudGV4dC1mdW5kaW5ne1xuICAgIHRleHQtYWxpZ246IGxlZnQ7XG4gIH1cblxuICAucGQtNjAge1xuICAgIHBhZGRpbmctdG9wOiA2MHB4O1xuICB9XG4gIC50ZXh0LWxlZnR7XG4gICAgdGV4dC1hbGlnbjogbGVmdDtcbiAgICBqdXN0aWZ5LWNvbnRlbnQ6IGxlZnQ7XG4gIH1cblxuICAvLyBWaWRlbyBzZWN0aW9uXG4gIC52aWRlby1zZWN0aW9ue1xuICAgIGJhY2tncm91bmQ6IHVybChcIi9hc3NldHMvaW1hZ2VzL2xhbmRpbmcvc21hcnRjaXRpemVuLXdvcmtzaG9wLmpwZ1wiKSBjZW50ZXIvY292ZXI7XG4gICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgIHRvcDogMHB4O1xuICAgIGhlaWdodDogMTAwdmg7XG4gICAgb3ZlcmZsb3c6IGhpZGRlbjtcbiAgICBib3JkZXI6IDNweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICAgIGJveC1zaXppbmc6IGJvcmRlci1ib3g7XG4gIH1cbiAgLm1vYmlsZW9ubHkge1xuICAgIGRpc3BsYXk6IG5vbmU7XG4gIH1cbiAgLmhlYWRlci1zZWN0aW9ue1xuICAgIG1hcmdpbjogNjBweCAxMCU7XG4gICAgbWF4LXdpZHRoOiAxMjAwcHg7XG5cbiAgfVxuICAuaGVhZGluZy1vdmVyLXZpZGVve1xuICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICB6LWluZGV4OiAyO1xuICAgIGJvdHRvbTogODBweDtcbiAgICBtYXgtd2lkdGg6IDYwMHB4O1xuICAgIHdpZHRoOiA1MCU7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogcmdiKDI1NSwgMjU1LCAyNTUpO1xuICAgIGJvcmRlcjogM3B4IHNvbGlkICRvZmZfYmxhY2s7XG4gICAgaW1ne1xuICAgICAgcGFkZGluZy1ib3R0b206NDBweDtcbiAgICB9XG4gIH1cbiAgLmhlYWRpbmctb3Zlci12aWRlby5yaWdodC1jYXJkIHtcbiAgICByaWdodDogMTAlO1xuICB9XG5cbiAgI2xlYXJuLW1vcmUge1xuICAgIG1hcmdpbjogMHB4IGF1dG8gNjBweCBhdXRvO1xuICAgIG1heC13aWR0aDoxMjAwcHg7XG4gICAgcGFkZGluZzogMCAxMCU7XG4gIH1cbiAgLy8gQmFubmVyIHNlY3Rpb25cblxuICAuYmFubmVyLXNlY3Rpb257XG4gICAgYmFja2dyb3VuZDogdXJsKFwiL2Fzc2V0cy9pbWFnZXMvbGFuZGluZy9TQ0stUGxhdGZvcm0uanBnXCIpIGNlbnRlci9jb3ZlcjtcbiAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gICAgaGVpZ2h0OiAxMDB2aDtcbiAgICB3aWR0aDogMTAwJTtcbiAgICBvdmVyZmxvdzogaGlkZGVuO1xuICAgIGJvcmRlcjogM3B4IHNvbGlkICRvZmZfYmxhY2s7XG4gICAgYm94LXNpemluZzogYm9yZGVyLWJveDtcbiAgfVxuXG4gIC8vIENhcmRzXG5cbiAgLmNhcmQge1xuICAgIGJveC1zaGFkb3c6IDEwcHggMTBweCAwcHggLTNweCAjRkZDMTAwLDEwcHggMTBweCAwcHggMHB4ICRvZmZfYmxhY2s7XG4gIH1cblxuICBAbWVkaWEgb25seSBzY3JlZW4gYW5kIChtYXgtd2lkdGg6IDEyMDBweCkge1xuICAgICNsZWFybi1tb3JlIHtcbiAgICAgIHBhZGRpbmc6IDA7XG4gICAgfVxuICAgIC5jYXJkIHtcbiAgICAgIGJveC1zaGFkb3c6IDBweCAwcHg7XG4gICAgfVxuICB9XG5cbiAgQG1lZGlhIG9ubHkgc2NyZWVuIGFuZCAobWF4LXdpZHRoOiAxMDEwcHgpIHtcbiAgICBoMXtcbiAgICAgIGZvbnQtc2l6ZTo0MnB4O1xuICAgICAgbGluZS1oZWlnaHQ6NDJweDtcbiAgICB9XG4gICAgaDJ7XG4gICAgICBmb250LXNpemU6MjhweDtcbiAgICAgIGxpbmUtaGVpZ2h0OjI4cHg7XG4gICAgfVxuICAgIGgze1xuICAgICAgZm9udC1zaXplOiAyNnB4O1xuICAgICAgbGluZS1oZWlnaHQ6IDI2cHg7XG4gICAgfVxuICAgIHB7XG4gICAgICBmb250LXNpemU6MThweDtcbiAgICAgIGxpbmUtaGVpZ2h0OjIycHg7XG4gICAgfVxuICAgIC5zYy1sb2dve1xuICAgICAgbGVmdDo1JTtcbiAgICB9XG4gICAgLmhlYWRpbmctb3Zlci12aWRlb3tcbiAgICAgIGltZ3tcbiAgICAgICAgcGFkZGluZy1ib3R0b206MDBweDtcbiAgICAgIH1cbiAgICB9XG4gICAgI2dldC15b3VyLWtpdCB7XG4gICAgICBzY3JvbGwtbWFyZ2luLXRvcDogODBweDtcbiAgICB9XG4gIH1cblxuICBAbWVkaWEgb25seSBzY3JlZW4gYW5kIChtYXgtd2lkdGg6IDk1MHB4KSB7XG4gICAgLm1vYmlsZW9ubHkge1xuICAgICAgZGlzcGxheTogYmxvY2s7XG4gICAgICBib3JkZXI6IDNweCBzb2xpZCAjMUMxQzFDO1xuICAgICAgYm9yZGVyLWJvdHRvbTogbm9uZTtcbiAgICB9XG4gICAgLm1vYmlsZW9ubHkuaW1nLXBsYXRmb3JtIHtcbiAgICAgIGJvcmRlci10b3A6IG5vbmU7XG4gICAgfVxuICAgIC5zYy1vZmYtY3RhLXBsYXRmb3JtIHtcbiAgICAgIGp1c3RpZnktc2VsZjogY2VudGVyO1xuICAgICAgYWxpZ24tc2VsZjogc3RhcnQ7XG4gICAgICBhbGlnbi1pdGVtczogc3RhcnQ7XG4gICAgICB0ZXh0LWFsaWduOiBsZWZ0O1xuICAgICAgZmxleC13cmFwOiB3cmFwO1xuICAgICAgLXdlYmtpdC10cmFuc2l0aW9uOiBhbGwgMC4zcyBlYXNlLWluLW91dDtcbiAgICAgIHRyYW5zaXRpb246IGFsbCAwLjNzIGVhc2UtaW4tb3V0O1xuICAgICAgfVxuICAgIC5uYXZpZ2F0aW9uLmFjdGl2ZSB7XG4gICAgICBwYWRkaW5nOiAxNnB4IDAlIDZweCAxMSU7XG4gICAgfVxuICAgIC5leHRlcm5hbC1saW5rcyB7XG4gICAgICBkaXNwbGF5OiBmbGV4O1xuICAgICAgZmxleC1kaXJlY3Rpb246IGNvbHVtbjtcbiAgICAgIH1cbiAgICAuaGFtYnVyZ2VyIHtcbiAgICAgIGRpc3BsYXk6IGJsb2NrO1xuICAgICAgY3Vyc29yOiBwb2ludGVyO1xuICAgICAgYWxpZ24tc2VsZjogY2VudGVyO1xuICAgICAgcGFkZGluZzogNnB4IDEwJSA2cHggMCU7XG4gICAgfVxuICAgIC5oYW1idXJnZXIuYWN0aXZlIHtcbiAgICAgIGRpc3BsYXk6IGJsb2NrO1xuICAgICAgY3Vyc29yOiBwb2ludGVyO1xuICAgICAgYWxpZ24tc2VsZjogY2VudGVyO1xuICAgICAgcGFkZGluZzogMTZweCAxMSUgNnB4IDAlO1xuICAgIH1cbiAgICAuaGFtYnVyZ2VyLmFjdGl2ZSAuYmFyOm50aC1jaGlsZCgyKSB7XG4gICAgICBvcGFjaXR5OiAwO1xuICAgIH1cbiAgICAuaGFtYnVyZ2VyLmFjdGl2ZSAuYmFyOm50aC1jaGlsZCgxKSB7XG4gICAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkoOHB4KSByb3RhdGUoNDVkZWcpO1xuICAgIH1cbiAgICAuaGFtYnVyZ2VyLmFjdGl2ZSAuYmFyOm50aC1jaGlsZCgzKSB7XG4gICAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkoLThweCkgcm90YXRlKC00NWRlZyk7XG4gICAgfVxuICAgIC5sYW5kaW5nLW1lbnUge1xuICAgICAgZGlzcGxheTogbm9uZTtcbiAgICAgIGp1c3RpZnktY29udGVudDogc3BhY2UtYmV0d2VlbjtcbiAgICAgIHdpZHRoOiAxMDAlO1xuICAgICAgaGVpZ2h0OiA0MCU7XG4gICAgICBwYWRkaW5nOiAwcHggMTAlO1xuICAgIH1cbiAgICAubGFuZGluZy1tZW51LmFjdGl2ZSB7XG4gICAgICBkaXNwbGF5OiBmbGV4O1xuICAgICAgYWxpZ24tc2VsZjogc3RhcnQ7XG4gICAgICAtd2Via2l0LXRyYW5zaXRpb246IGFsbCAuM3MgZWFzZS1pbi1vdXQ7XG4gICAgICB0cmFuc2l0aW9uOiBhbGwgLjNzIGVhc2UtaW4tb3V0O1xuICAgICAgZmxleC1kaXJlY3Rpb246IGNvbHVtbjtcbiAgICAgIGp1c3RpZnktY29udGVudDogc3BhY2UtYmV0d2VlbjtcbiAgICAgIHBhZGRpbmctYm90dG9tOiAxMCU7XG4gICAgfVxuICAgIC5sYW5kaW5nLW1lbnUgLm1kLWJ1dHRvbi5sYW5kaW5nLW1lbnUtYnRuIHtcbiAgICAgIHRleHQtYWxpZ246IGxlZnQ7XG4gICAgICBwYWRkaW5nOiAwcHggMHB4O1xuICAgIH1cbiAgIC5sYW5kaW5nLW1lbnUgLmJ0bi1yb3VuZC1uZXcubWQtYnV0dG9uLmJ0bi1raXQge1xuICAgICAgYWxpZ24tc2VsZjogZW5kO1xuICAgICAganVzdGlmeS1zZWxmOiBlbmQ7XG4gICAgfVxuICAgIC5zYy1vZmYtY3RhLXBsYXRmb3JtLmFjdGl2ZSB7XG4gICAgICBoZWlnaHQ6IDEwMCU7XG4gICAgICB3aWR0aDogMTAwJTtcbiAgICAgIGp1c3RpZnktY29udGVudDogc3BhY2UtYmV0d2VlbjtcbiAgICAgIGFsaWduLWNvbnRlbnQ6IHNwYWNlLWJldHdlZW47XG4gICAgICAtd2Via2l0LXRyYW5zaXRpb246IGFsbCAuM3MgZWFzZS1pbi1vdXQ7XG4gICAgICB0cmFuc2l0aW9uOiBhbGwgLjNzIGVhc2UtaW4tb3V0O1xuICAgICAgbWFyZ2luOiAwJTtcbiAgICB9XG4gICAgLnNjLWxvZ297XG4gICAgICBwYWRkaW5nOiAxMHB4IDBweDtcbiAgICB9XG4gICAgI2dldC15b3VyLWtpdCB7XG4gICAgICBzY3JvbGwtbWFyZ2luLXRvcDogODVweDtcbiAgICB9XG4gICAgLmxheW91dC14cy1jb2x1bW4ge1xuICAgICBmbGV4LWRpcmVjdGlvbjogY29sdW1uO1xuICAgIH1cbiAgICAubGF5b3V0LXhzLWNvbHVtbiA+IC5mbGV4LXhzLTEwMCB7XG4gICAgICBmbGV4OiAxIDEgMTAwJTtcbiAgICAgIG1heC13aWR0aDogMTAwJTtcbiAgICAgIG1heC1oZWlnaHQ6IDEwMCU7XG4gICAgICBib3gtc2l6aW5nOiBib3JkZXItYm94O1xuICAgIH1cbiAgICAuYm9yZGVyLXhzLXRvcHtcbiAgICAgIGJvcmRlci10b3A6IDNweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICAgICAgYm9yZGVyLWxlZnQ6IDNweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICAgICAgYm9yZGVyLXJpZ2h0OiAzcHggc29saWQgJG9mZl9ibGFjaztcbiAgICAgIGJvcmRlci1ib3R0b206IDNweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICAgIH1cbiAgICAuYm9yZGVyLXhzLWJvdHRvbXtcbiAgICAgIGJvcmRlci10b3A6IHVuc2V0O1xuICAgICAgYm9yZGVyLWxlZnQ6IDNweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICAgICAgYm9yZGVyLXJpZ2h0OiAzcHggc29saWQgJG9mZl9ibGFjaztcbiAgICAgIGJvcmRlci1ib3R0b206M3B4IHNvbGlkICRvZmZfYmxhY2s7XG4gICAgfVxuICAgIC5mbGV4LW9yZGVyLXhzLTEge1xuICAgICAgb3JkZXI6IDEgIWltcG9ydGFudDtcbiAgICB9XG4gICAgLmZsZXgtb3JkZXIteHMtMiB7XG4gICAgICBvcmRlcjogMiAhaW1wb3J0YW50O1xuICAgIH1cbiAgICAubGF5b3V0LXhzLWNvbHVtbiB7XG4gICAgICBib3gtc2l6aW5nOiBib3JkZXItYm94O1xuICAgICAgZGlzcGxheTogZmxleDtcbiAgICB9XG4gICAgLmhlYWRpbmctb3Zlci12aWRlb3tcbiAgICAgIHBvc2l0aW9uOiBzdGF0aWM7XG4gICAgICBib3R0b206IDBweDtcbiAgICAgIHdpZHRoOiAxMDAlO1xuICAgICAgbWFyZ2luOiAwcHg7XG4gICAgICBib3JkZXItYm90dG9tOiAwcHg7XG4gICAgICBib3JkZXItbGVmdDogMHB4O1xuICAgICAgYm9yZGVyLXJpZ2h0OiAwcHg7XG4gICAgICBtYXgtd2lkdGg6IDEwMCU7XG4gICAgfVxuICAgIC52aWRlby1zZWN0aW9uIC5oZWFkaW5nLW92ZXItdmlkZW97XG4gICAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgfVxuICAgIC5oZWFkaW5nLW92ZXItdmlkZW8ucmlnaHQtY2FyZCB7XG4gICAgICByaWdodDogMHB4O1xuICAgIH1cbiAgICAuaGVhZGVyLXNlY3Rpb24ge1xuICAgICAgbWFyZ2luOiAwcHg7XG4gICAgfVxuICAgIC5iYW5uZXItc2VjdGlvbiB7XG4gICAgICBwb3NpdGlvbjogc3RhdGljO1xuICAgICAgZGlzcGxheTogZmxleDtcbiAgICAgIGZsZXgtZGlyZWN0aW9uOiBjb2x1bW4tcmV2ZXJzZTtcbiAgICAgIGhlaWdodDogYXV0bztcbiAgICAgIGJvcmRlcjogbm9uZTtcbiAgICB9XG4gICAgLnAtaGVhZGVye1xuICAgICAgZm9udC1zaXplOjE5cHg7XG4gICAgfVxuXG4gICAgLnRpbGUtaW1hZ2UudGlsZSB7XG4gICAgICBtaW4taGVpZ2h0OjUwMHB4O1xuICAgIH1cbiAgICAudGV4dC1sZWZ0e1xuICAgICAgdGV4dC1hbGlnbjogY2VudGVyO1xuICAgICAganVzdGlmeS1jb250ZW50OiBjZW50ZXI7XG4gICAgfVxuICAgIC5iYW5uZXItc2VjdGlvbntcbiAgICAgIGhlaWdodDogMTAwJTtcbiAgICB9XG4gICAgLmJ0bi1qdXN0aWZ5LWxlZnR7XG4gICAgICBtYXJnaW4tbGVmdDogOHB4O1xuICAgIH1cbiAgICAuY2FyZC10ZXh0IHtcbiAgICAgIHRleHQtYWxpZ246IGNlbnRlcjtcbiAgICAgIGJvcmRlci1sZWZ0OiAzcHggc29saWQgJG9mZl9ibGFjaztcbiAgICAgIGJvcmRlci1yaWdodDogM3B4IHNvbGlkICRvZmZfYmxhY2s7XG4gICAgICBhbGlnbi1zZWxmOiBzdHJldGNoO1xuICAgIH1cbiAgICAudmlkZW8tc2VjdGlvbiAuY2FyZC10ZXh0IHtcbiAgICAgIGJvcmRlci1sZWZ0OiBub25lO1xuICAgICAgYm9yZGVyLXJpZ2h0OiBub25lO1xuICAgIH1cblxuICAgIC8vIE1vYmlsZSByZW1vdmUgcGFkZGluZyBhbmQgZXh0cmEgbGluZXNcblxuICAgIC52aWRlby1zZWN0aW9uIHtcbiAgICAgIGJvcmRlci1ib3R0b206IDBweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICAgIH1cbiAgICAucGQtNjAge1xuICAgICAgcGFkZGluZzogMHB4O1xuICAgIH1cbiAgICAuYm9yZGVyLWJsYWNrIHtcbiAgICAgIGJvcmRlci10b3A6IDBweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICAgIH1cbiAgICAubWItMTAge1xuICAgICAgbWFyZ2luLWJvdHRvbTogMHB4O1xuICAgIH1cbiAgICAuYmFubmVyLXNlY3Rpb24ge1xuICAgICAgYm9yZGVyLWJvdHRvbTogMHB4IHNvbGlkICRvZmZfYmxhY2s7XG4gICAgICBib3JkZXItdG9wOiAwcHggc29saWQgJG9mZl9ibGFjaztcbiAgICB9XG4gICAgI2xlYXJuLW1vcmUge1xuICAgICAgbWFyZ2luOiAwcHggYXV0byAwcHggYXV0b1xuICAgIH1cbiAgICAjb3Blbi1wbGF0Zm9ybSB7XG4gICAgICBmbGV4LWRpcmVjdGlvbjogY29sdW1uLXJldmVyc2U7XG4gICAgICBib3JkZXItdG9wOiAzcHggc29saWQgJG9mZl9ibGFjaztcbiAgICAgIGJvcmRlci1ib3R0b206IDNweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICAgIH1cbiAgICAjb3Blbi1wbGF0Zm9ybSAuZmxleC14cy0xMDAge1xuICAgICAgZmxleC1kaXJlY3Rpb246IGNvbHVtbjtcbiAgICAgIGJvcmRlci1ib3R0b206IDBweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICAgIH1cbiAgICAubXItMzAge1xuICAgICAgbWFyZ2luLXJpZ2h0OiAwcHg7XG4gICAgfVxuICAgIC5uZXctbGFuZGluZy1wYWdlIHtcbiAgICAgIHBhZGRpbmctYm90dG9tOiAwcHg7XG4gICAgICBiYWNrZ3JvdW5kLXNpemU6IDBweDtcbiAgICB9XG4gICAgLmJhbm5lci1zZWN0aW9ue1xuICAgICAgYmFja2dyb3VuZDogbm9uZTtcbiAgICB9XG4gIH1cblxuICAvLyBMZXNzIHRoZW4gNzUwcHhcbiAgQG1lZGlhIG9ubHkgc2NyZWVuIGFuZCAobWF4LXdpZHRoOiA3NTBweCkge1xuICAgIGgxe1xuICAgICAgZm9udC1zaXplOjMwcHg7XG4gICAgfVxuICAgIGgye1xuICAgICAgZm9udC1zaXplOjI1cHg7XG4gICAgfVxuICAgIC50ZXh0LWZ1bmRpbmd7XG4gICAgICB0ZXh0LWFsaWduOiBjZW50ZXI7XG4gICAgfVxuICB9XG5cbiAgQG1lZGlhIG9ubHkgc2NyZWVuIGFuZCAobWF4LXdpZHRoOiA2MDBweCkge1xuICAgIGgxe1xuICAgICAgZm9udC1zaXplOjI4cHg7XG4gICAgICBsaW5lLWhlaWdodDozM3B4O1xuICAgIH1cbiAgICBoMntcbiAgICAgIGZvbnQtc2l6ZToyOHB4O1xuICAgIH1cbiAgICBwe1xuICAgICAgZm9udC1zaXplOjE5cHg7XG4gICAgfVxuICAgIC5wLWhlYWRlcntcbiAgICAgIGZvbnQtc2l6ZToxOXB4O1xuICAgIH1cbiAgICAucC02MHtcbiAgICAgIHBhZGRpbmc6IDEwcHggMTBweCAhaW1wb3J0YW50O1xuICAgIH1cbiAgICAubXktMjB7XG4gICAgICBtYXJnaW4tdG9wOiAxMHB4ICFpbXBvcnRhbnQ7XG4gICAgICBtYXJnaW4tYm90dG9tOiAxMHB4ICFpbXBvcnRhbnQ7XG4gICAgfVxuXG4gICAgLnRpbGUtaW1hZ2UudGlsZXtcbiAgICAgIG1pbi1oZWlnaHQ6MzAwcHg7XG4gICAgfVxuICAgIC50aWxle1xuICAgICAgcGFkZGluZzogNTBweCAzMHB4O1xuICAgIH1cbiAgfVxufSIsIi8qKlxuICogIE5BVkJBUlxuICovXG5cbi5uYXZiYXJfY29udGFpbmVyIHtcbiAgYXtcbiAgICAmOmhvdmVyLCAmOmFjdGl2ZXtcbiAgICAgIGNvbG9yOiAkZm9udF9jb2xvcl93aGl0ZVxuICAgIH1cbiAgfVxuXG4gIC5tZC1idXR0b24ubmF2YmFyX2hpZ2hsaWdodF9idXR0b24ge1xuICAgIGJvcmRlcjogMnB4IHNvbGlkICR5ZWxsb3c7XG4gICAgY29sb3I6ICR5ZWxsb3c7XG4gICAgYm9yZGVyLXJhZGl1czogNTBweDtcbiAgICAvL3BhZGRpbmc6IDlweCAxMXB4O1xuICAgIC8vd2lkdGg6IDEwOHB4O1xuICAgIHBhZGRpbmctdG9wOiAuMDFyZW07XG4gICAgcGFkZGluZy1ib3R0b206IC4wMXJlbTtcbiAgICB0cmFuc2l0aW9uOiBjb2xvciAwLjRzIGN1YmljLWJlemllcigwLjI1LCAwLjgsIDAuMjUsIDEpLCBiYWNrZ3JvdW5kLWNvbG9yIDAuNHMgY3ViaWMtYmV6aWVyKDAuMjUsIDAuOCwgMC4yNSwgMSk7XG4gICAgJjpob3ZlciwgJjphY3RpdmUge1xuICAgICAgYmFja2dyb3VuZC1jb2xvcjokeWVsbG93ICFpbXBvcnRhbnQ7XG4gICAgICBjb2xvcjogJG9mZl9ibGFjayAhaW1wb3J0YW50O1xuICAgIH1cbiAgfVxuXG4gIC5sb2dvX2xpbmsge1xuICAgIHRleHQtYWxpZ246IGNlbnRlcjtcblxuICAgIC5sb2dvX2ljb24ge1xuICAgICAgaGVpZ2h0OiAzMnB4O1xuICAgICAgd2lkdGg6IDMycHg7XG4gICAgfVxuICB9XG5cbiAgLm5hdl9pY29uIHtcbiAgICBtYXJnaW4tcmlnaHQ6IDRweDtcbiAgICBoZWlnaHQ6IDE0cHggIWltcG9ydGFudDtcbiAgICB3aWR0aDogMTRweCAhaW1wb3J0YW50O1xuICAgIHN2ZyB7XG4gICAgICBmaWxsOiAkeWVsbG93O1xuICAgIH1cbiAgfVxuXG4gIC5uYXZiYXJfc2lnbnVwX2J1dHRvbiB7XG4gICAgLm1kLWJ1dHRvbiB7XG4gICAgICBwYWRkaW5nOiAycHggMHB4O1xuICAgICAgYm9yZGVyOiAycHggc29saWQgJHllbGxvdztcbiAgICAgIGNvbG9yOiAkeWVsbG93O1xuICAgICAgYm9yZGVyLXJhZGl1czogNTBweDtcbiAgICAgIHdpZHRoOiA3OXB4O1xuICAgIH1cbiAgICAubWQtYnV0dG9uOmhvdmVyIHtcbiAgICAgIGJhY2tncm91bmQtY29sb3I6ICR5ZWxsb3c7XG4gICAgICBjb2xvcjogJG9mZl9ibGFjaztcbiAgICB9XG4gIH1cblxufVxuXG4ubmF2YmFyX2F2YXRhcl9pY29uIHtcbiAgd2lkdGg6IDM4cHg7XG4gIGhlaWdodDogMzhweDtcbiAgYm9yZGVyLXJhZGl1czogMTlweDtcbn1cblxuLy8gTWFrZSBzdXJlIHRoZSBEcm9wZG93biBzdGF5cyBiZWxvdyB0aGUgYnV0dG9uIHB1c2hlZFxuLm1kLW9wZW4tbWVudS1jb250YWluZXIubWQtYWN0aXZle1xuICBtYXJnaW4tdG9wOiA1MHB4ICFpbXBvcnRhbnQ7XG59XG5cbiIsIi8vIC8qKlxuLy8gICogIFNFQVJDSCBDT01QT05FTlRcbi8vICAqL1xuXG5cbiNzZWFyY2ggaW5wdXQge1xuICAgIGJhY2tncm91bmQ6IHVybCguLi9hc3NldHMvaW1hZ2VzL3NlYXJjaF9pY29uX2JsYWNrLnN2Zyk7XG4gICAgYmFja2dyb3VuZC1yZXBlYXQ6IG5vLXJlcGVhdDtcbiAgICBiYWNrZ3JvdW5kLXBvc2l0aW9uOiAxMHB4IDEycHg7XG4gICAgYmFja2dyb3VuZC1zaXplOiAxOHB4O1xuICAgIHBhZGRpbmc6IDBweCAzNXB4O1xufVxuXG4uc2VhcmNoX3Jlc3VsdHMge1xuICBkaXNwbGF5OiBmbGV4O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICAucmVzdWx0X2ljb24ge1xuICAgIGZsZXg6IDAgMCAxOHB4O1xuICAgIG1hcmdpbi1yaWdodDogMTJweDtcbiAgICBoZWlnaHQ6IDE4cHg7XG4gIH1cbiAgLnJlc3VsdF9uYW1lIHtcbiAgICBtYXJnaW4tcmlnaHQ6IDEwcHg7XG4gIH1cbiAgLnJlc3VsdF9sb2NhdGlvbiB7XG4gICAgY29sb3I6ICRibHVlO1xuICB9XG59XG4iLCIvKlxuICBDT05UQUlORVJTIFVTRUQgT04gU0hPVyBLSVQgU1RBVEVcbiAqL1xuXG4ub3Zlcl9tYXB7XG4gIG1pbi1oZWlnaHQ6IDEyMHB4O1xufVxuXG4ua2l0X2ZpeGVkIHtcbiAgei1pbmRleDogJG1lZGl1bS1wcmlvcml0eTtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xufVxuXG4ua2l0X2RhdGEge1xuICB3aWR0aDoxMDAlO1xuICBtYXJnaW4tdG9wOiAzNzZweDtcblxuICBzZWN0aW9uLm92ZXJsYXl7XG4gICAgd2lkdGg6MTAwJTtcbiAgICBoZWlnaHQ6MTAwJTtcbiAgICBwb3NpdGlvbjphYnNvbHV0ZTtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiB3aGl0ZTtcbiAgICB6LWluZGV4OiAzO1xuICAgIGNvbG9yOiAkb2ZmX2JsYWNrO1xuICAgIHRleHQtYWxpZ246Y2VudGVyO1xuICAgIGZvbnQtc2l6ZToyMHB4O1xuICAgIGZvbnQtd2VpZ2h0OiBsaWdodGVyO1xuXG4gICAgaDF7XG4gICAgICBjb2xvcjogJG9mZl9ibGFjaztcbiAgICAgIGZvbnQtc2l6ZTogMS44ZW07XG4gICAgICBtYXJnaW46MDtcbiAgICB9XG4gICAgcHtcbiAgICAgIGNvbG9yOiAkdGVyY2lhcnlfY29sb3JfbGlnaHQ7XG4gICAgICBtYXJnaW46IDAuNWVtO1xuICAgIH1cbiAgfVxuXG4gIC5oaW50e1xuICAgIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xuICAgIHRleHQtYWxpZ246Y2VudGVyO1xuICAgIGNvbG9yOiRmb250X2NvbG9yX2JsdWVfZ3JleTtcblxuICAgIG1hcmdpbi10b3A6MHB4O1xuXG4gICAgei1pbmRleDogMTA7XG4gICAgcG9zaXRpb246YWJzb2x1dGU7XG4gICAgaGVpZ2h0OjEwMCU7XG4gICAgd2lkdGg6MTAwJTtcbiAgICA+IHAge1xuICAgICAgY29sb3I6ICRncmV5X2RhcmtlcjtcbiAgICB9XG4gICAgLm1kLWJ1dHRvbntcbiAgICAgIG1hcmdpbi10b3A6MjBweDtcbiAgICB9XG4gIH1cbn1cblxuLmtpdF9kYXRhLCAua2l0VGFnc19fc2VjdGlvbntcbiAgLnNoYWRvd3tcbiAgICB0b3A6IDBweDtcbiAgICBoZWlnaHQ6IDFweDtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiByZ2JhKDAsMCwwLDAuMSk7XG4gICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgIGJveC1zaGFkb3c6IDBweCAtMXB4IDZweCAwIHJnYmEoMCwwLDAsMC42NSk7XG4gICAgei1pbmRleDogMjtcbiAgfVxufVxuLmtpdF90YWdzLXNlbGVjdC1oZWFkZXIge1xuICBoZWlnaHQ6IDQ4cHg7XG4gIGRpc3BsYXk6IGZsZXg7XG59XG4ua2l0X3RhZ3MtaGVhZGVyLXNlYXJjaGJveCB7XG4gIGJvcmRlcjogbm9uZTtcbiAgYm9yZGVyLWJvdHRvbTogMXB4IHNvbGlkICNjY2M7XG4gIHBhZGRpbmctbGVmdDogMTJweDtcbiAgaGVpZ2h0OiAxMDAlO1xuICB3aWR0aDogMTAwJTtcbn1cbiIsIi8qXG49PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuS0lUIENPTU1FTlRTIE1PRFVMRVxuPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbiovXG4ua2l0X2NvbW1lbnRzIHtcbiAgcGFkZGluZy1ib3R0b206IDEwMHB4O1xufVxuXG4vKlxuICBDSEFSVCBDT01QT05FTlRcbiovXG5cbi5jaGFydF9jb250YWluZXIge1xuICAvKm1pbi13aWR0aDogMzQwcHg7Ki9cbiAgbWluLWhlaWdodDogMzEwcHg7XG4gIC8qd2lkdGg6IDEwMCU7Ki9cbiAgLypoZWlnaHQ6IDkwJTsqL1xuICBwb3NpdGlvbjogcmVsYXRpdmU7XG5cbiAgLmNoYXJ0X2xpbmUge1xuICAgIGZpbGw6IG5vbmU7XG4gICAgc3Ryb2tlLXdpZHRoOiAycHg7XG4gIH1cbiAgLmNoYXJ0X2FyZWEge1xuICAgIG9wYWNpdHk6IDAuMTtcbiAgfVxuXG4gIC5heGlzIHBhdGgsLmF4aXMgbGluZSB7XG4gICAgZmlsbDogbm9uZTtcbiAgICBzdHJva2U6ICRncmV5X2RhcmtlcjtcbiAgICBzdHJva2Utd2lkdGg6IDE7XG4gICAgc2hhcGUtcmVuZGVyaW5nOiBjcmlzcEVkZ2VzO1xuICAgIGRpc3BsYXk6IG5vbmU7XG4gIH1cbiAgLmF4aXMge1xuICAgIGZvbnQtc2l6ZTogMTBweDtcbiAgfVxuICAuYXhpcy55X2xlZnQge1xuICAgIGZpbGw6ICRncmV5O1xuICB9XG4gIC5heGlzLnlfcmlnaHQge1xuICAgIGZpbGw6ICRncmV5O1xuICB9XG4gIC5heGlzLngge1xuICAgIGZpbGw6ICRncmV5O1xuICB9XG4gIC5ncmlkIHtcbiAgICAudGljayB7XG4gICAgICBzdHJva2U6ICRncmV5O1xuICAgICAgc3Ryb2tlLW9wYWNpdHk6IDAuNjtcbiAgICAgIHNoYXBlLXJlbmRlcmluZzogY3Jpc3BFZGdlcztcbiAgICB9XG4gICAgcGF0aCB7XG4gICAgICBzdHJva2Utd2lkdGg6IDA7XG4gICAgfVxuICB9XG4gIC5vdmVybGF5IHtcbiAgICBmaWxsOiBub25lO1xuICAgIHBvaW50ZXItZXZlbnRzOiBhbGw7XG4gIH1cblxuICAuZm9jdXMgY2lyY2xlIHtcbiAgICBmaWxsOiByZ2IoMywgMzcsIDQ1KTtcbiAgICBzdHJva2Utd2lkdGg6IDJweDtcbiAgfVxuICAudGV4dF9ob3Zlcl9jb250YWluZXIge1xuICAgIGJvcmRlcjogMXB4IHNvbGlkICRncmV5X2RhcmtlcjtcbiAgfVxuICAucG9wdXBfdmFsdWUge1xuICAgIGZpbGw6ICRiYWNrZ3JvdW5kX2JsdWVfZGFya19jaGFydDtcbiAgICBjb2xvcjogJGJhY2tncm91bmRfYmx1ZV9kYXJrX2NoYXJ0O1xuICAgIGZvbnQtc2l6ZTogMThweDtcbiAgICBmb250LXdlaWdodDogYm9sZDtcbiAgfVxuICAucG9wdXBfZGF0ZSB7XG4gICAgZmlsbDogJGdyZXlfZGFya2VyO1xuICAgIGZvbnQtc2l6ZTogMTBweDtcbiAgICBmb250LXdlaWdodDogYm9sZDtcbiAgfVxuICAucG9wdXBfdW5pdCB7XG4gICAgZmlsbDogJGdyZXk7XG4gICAgbWFyZ2luLWxlZnQ6IDEwcHg7XG4gIH1cbn1cblxuLnN0aWNrTmF2IHtcbiAgcG9zaXRpb246IGZpeGVkO1xuICB0b3A6IDBweDtcbn1cblxuLm92ZXJsYXkta2l0aW5mb3tcbiAgYmFja2dyb3VuZDogd2hpdGU7XG4gIGhlaWdodDogMTAwJTtcbiAgd2lkdGg6IDEwMCU7XG4gIHBvc2l0aW9uOmFic29sdXRlO1xuICB6LWluZGV4Ojk5OTtcbn1cblxuXG4vKlxuICBLSVQgTUVOVSBNT0RVTEVcbiAqL1xuc2VjdGlvbiAua2l0X21lbnUge1xuICB3aWR0aDogMTAwJTtcbiAgei1pbmRleDogJG1vcmVfcHJpb3JpdHk7XG4gIGJhY2tncm91bmQtY29sb3I6ICRncmV5X2xpZ2h0ZXI7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcblxuICAuY29udGFpbmVyIHtcbiAgICB3aWR0aDogMTAwJTtcbiAgICBoZWlnaHQ6IDMzcHg7XG4gIH1cblxuICAua2l0X3RpbWUge1xuICAgIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgICAvL2xlZnQ6IC00MHB4O1xuICAgIHRleHQtYWxpZ246IGNlbnRlcjtcbiAgICBjb2xvcjogJGdyZXlfZGFya2VyO1xuICAgIGxldHRlci1zcGFjaW5nOiAwLjVweDtcbiAgICBmb250LXNpemU6IDE0cHg7XG4gICAgdGV4dC10cmFuc2Zvcm06IHVwcGVyY2FzZTtcbiAgfVxuICAua2l0X3VzZXIge1xuICAgIG1hcmdpbi1sZWZ0OjIwcHg7XG4gICAgLy9mbG9hdDogbGVmdDtcbiAgICBsaW5lLWhlaWdodDogMzBweDtcbiAgICAvL21hcmdpbjogMCAzMnB4IDAgMjBweDtcbiAgICBjb2xvcjogJGdyZXlfZGFya2VyO1xuICAgIGZvbnQtd2VpZ2h0OiBib2xkO1xuICAgIGZvbnQtc2l6ZTogMTNweDtcblxuICAgIGltZyB7XG4gICAgICBib3JkZXItcmFkaXVzOjUwJTtcbiAgICAgIGZsb2F0OmxlZnQ7XG4gICAgICB3aWR0aDogMThweDtcbiAgICAgIGhlaWdodDogMThweDtcbiAgICAgIG1hcmdpbi1yaWdodDogOHB4O1xuICAgICAgbWFyZ2luLXRvcDogNXB4O1xuICAgIH1cblxuICAgIGEge1xuICAgICAgY29sb3I6ICRncmV5X2RhcmtlcjtcbiAgICAgIHRleHQtZGVjb3JhdGlvbjogbm9uZTtcbiAgICB9XG4gIH1cbiAgLmtpdF9uYW1lIHtcbiAgICBtYXJnaW4tbGVmdDozMHB4O1xuICAgIC8vZmxvYXQ6IGxlZnQ7XG4gICAgbGluZS1oZWlnaHQ6IDMwcHg7XG4gICAgZm9udC1zaXplOiAxM3B4O1xuICAgIGZvbnQtd2VpZ2h0OiBib2xkO1xuICAgIGNvbG9yOiAkZ3JleV9kYXJrZXI7XG4gICAgLy9tYXJnaW46IDAgMzJweCAwIDA7XG5cbiAgICBtZC1pY29uIHtcbiAgICAgIGZsb2F0OmxlZnQ7XG4gICAgICB3aWR0aDogMTRweDtcbiAgICAgIGhlaWdodDogMTRweDtcbiAgICAgIG1hcmdpbi1yaWdodDogOHB4O1xuICAgICAgbWFyZ2luLXRvcDogNHB4O1xuICAgICAgc3ZnIHsgZmlsbDogJHllbGxvdzsgfVxuICAgIH1cbiAgICBzcGFuIHtcbiAgICAgIHZlcnRpY2FsLWFsaWduOiBtaWRkbGU7XG4gICAgfVxuICB9XG5cbiAgLmtpdF9iYXR0ZXJ5IHtcbiAgICBtYXJnaW4tbGVmdDozMHB4O1xuICAgIC8vZmxvYXQ6IGxlZnQ7XG4gICAgbGluZS1oZWlnaHQ6IDMwcHg7XG4gICAgY29sb3I6ICRncmV5X2RhcmtlcjtcbiAgICBmb250LXNpemU6IDEycHg7XG4gICAgZm9udC13ZWlnaHQ6IGJvbGQ7XG5cbiAgICBtZC1pY29uIHtcbiAgICAgIG1hcmdpbi1ib3R0b206IDJweDtcbiAgICAgIHdpZHRoOiAxNnB4O1xuICAgICAgbWFyZ2luLXJpZ2h0OiA4cHg7XG4gICAgICBzdmcge1xuICAgICAgICBmaWxsOiAkZ3JleV9kYXJrZXI7XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgLmtpdC1zaG93LXJhd3tcbiAgICBtYXJnaW4tcmlnaHQ6IDZweDtcbiAgfVxuXG4gIC5raXQtcmF3LXRvZ2dsZXtcbiAgICB2ZXJ0aWNhbC1hbGlnbjogbWlkZGxlO1xuICB9XG5cbiAgLypTaG93IHJhdyBUb2dnbGUqL1xuICAuc3dpdGNoIHtcbiAgICB2ZXJ0aWNhbC1hbGlnbjogbWlkZGxlO1xuICAgIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgICBkaXNwbGF5OiBpbmxpbmUtYmxvY2s7XG4gICAgd2lkdGg6IDMwcHg7XG4gICAgaGVpZ2h0OiAxNXB4O1xuICB9XG5cbiAgLnN3aXRjaCBpbnB1dCB7IFxuICAgIG9wYWNpdHk6IDA7XG4gICAgd2lkdGg6IDA7XG4gICAgaGVpZ2h0OiAwO1xuICB9XG5cbiAgLnNsaWRlciB7XG4gICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgIHZlcnRpY2FsLWFsaWduOiBtaWRkbGU7XG4gICAgY3Vyc29yOiBwb2ludGVyO1xuICAgIHRvcDogMDtcbiAgICBsZWZ0OiAwO1xuICAgIHJpZ2h0OiAwO1xuICAgIGJvdHRvbTogMDtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjY2NjO1xuICAgIC13ZWJraXQtdHJhbnNpdGlvbjogLjRzO1xuICAgIHRyYW5zaXRpb246IC40cztcbiAgfVxuXG4gIC5zbGlkZXI6YmVmb3JlIHtcbiAgICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gICAgY29udGVudDogXCJcIjtcbiAgICBoZWlnaHQ6IDExcHg7XG4gICAgd2lkdGg6IDExcHg7XG4gICAgbGVmdDogM3B4O1xuICAgIGJvdHRvbTogMnB4O1xuICAgIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xuICAgIC13ZWJraXQtdHJhbnNpdGlvbjogLjRzO1xuICAgIHRyYW5zaXRpb246IC40cztcbiAgfVxuXG4gIGlucHV0OmNoZWNrZWQgKyAuc2xpZGVyIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkeWVsbG93O1xuICB9XG5cbiAgaW5wdXQ6Zm9jdXMgKyAuc2xpZGVyIHtcbiAgICBib3gtc2hhZG93OiAwIDAgMXB4ICMyMTk2RjM7XG4gIH1cblxuICBpbnB1dDpjaGVja2VkICsgLnNsaWRlcjpiZWZvcmUge1xuICAgIC13ZWJraXQtdHJhbnNmb3JtOiB0cmFuc2xhdGVYKDE0cHgpO1xuICAgIC1tcy10cmFuc2Zvcm06IHRyYW5zbGF0ZVgoMTRweCk7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVYKDE0cHgpO1xuICB9XG5cbiAgLyogUm91bmRlZCBzbGlkZXIgKi9cbiAgLnNsaWRlci5yb3VuZCB7XG4gICAgYm9yZGVyLXJhZGl1czogMzRweDtcbiAgfVxuXG4gIC5zbGlkZXIucm91bmQ6YmVmb3JlIHtcbiAgICBib3JkZXItcmFkaXVzOiA1MCU7XG4gIH1cblxuICAua2l0LXNob3ctcmF3LXRleHR7XG4gICAgdmVydGljYWwtYWxpZ246IG1pZGRsZTtcbiAgICBjb2xvcjogJGdyZXlfZGFya2VyO1xuICAgIGZvbnQtc2l6ZTogMTJweDtcbiAgICBmb250LXdlaWdodDogYm9sZDtcbiAgfVxuXG4gIC5raXRfbmF2YmFyIHtcbiAgICBtYXJnaW4tcmlnaHQ6IDIwcHg7XG5cbiAgICAvLyBJY29ucyBhcmUgaW5zaWRlIG1kLWJ1dHRvblxuICAgIG1kLWljb24ge1xuICAgICAgd2lkdGg6IDE0cHg7XG4gICAgICBzdmcge1xuICAgICAgICBmaWxsOiAkZ3JleV9kYXJrZXI7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLm1kLWJ1dHRvbiB7XG4gICAgICBtYXJnaW46MDtcbiAgICB9XG5cbiAgICBhLm1kLWJ1dHRvbjpob3ZlciwgLmJ1dHRvbl9hY3RpdmUgIHtcbiAgICAgIGJvcmRlci1ib3R0b206IDJweCBzb2xpZCAkb2ZmX2JsYWNrO1xuICAgIH1cblxuICB9XG5cbn1cblxuLmtpdF9vdmVydmlldyB7XG4gIHdpZHRoOiAxMDAlO1xuICBoZWlnaHQ6IDk2cHg7XG4gIHotaW5kZXg6ICRtZWRpdW1fcHJpb3JpdHk7XG4gIGJhY2tncm91bmQtY29sb3I6ICRncmV5X2xpZ2h0ZXN0O1xuICBwb3NpdGlvbjogcmVsYXRpdmU7XG5cbiAgLmJ1dHRvbl9zY3JvbGwge1xuICAgIG1hcmdpbjogMHB4O1xuICAgIC8vcG9zaXRpb246IGFic29sdXRlO1xuICAgIC8vdG9wOiAwO1xuICAgIC8vd2lkdGg6IDYwcHg7XG4gICAgLy9oZWlnaHQ6IDk2cHg7XG4gICAgYmFja2dyb3VuZDogI2NjYztcbiAgfVxuXG4gIC5idXR0b25fc2Nyb2xsX2xlZnQge1xuICAgIGxlZnQ6IDA7XG4gIH1cblxuICAuYnV0dG9uX3Njcm9sbF9yaWdodCB7XG4gICAgcmlnaHQ6IDA7XG4gIH1cblxuICAuc2Vuc29yc19jb250YWluZXIge1xuICAgIHdpZHRoOiA5MCU7XG4gICAgaGVpZ2h0OiA5NnB4O1xuICAgIG1hcmdpbjogMCBhdXRvO1xuICAgIG92ZXJmbG93LXg6IGhpZGRlbjtcbiAgICBvdmVyZmxvdy15OiBoaWRkZW47XG4gIH1cblxuICBAbWVkaWEgKHBvaW50ZXI6IGNvYXJzZSkge1xuICAgIC5zZW5zb3JzX2NvbnRhaW5lciB7XG4gICAgICBvdmVyZmxvdy14OiBhdXRvO1xuICAgIH1cbiAgfVxuXG4gIC5zZW5zb3JfY29udGFpbmVyIHtcbiAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gICAgY29sb3I6ICRncmV5X2RhcmtlcjtcbiAgICBkaXNwbGF5OiBpbmxpbmUtdGFibGU7XG4gICAgY3Vyc29yOiBwb2ludGVyO1xuICAgIHRleHQtYWxpZ246IGNlbnRlcjtcbiAgICBvdXRsaW5lOm5vbmU7XG5cbiAgICBwe1xuICAgICAgbWFyZ2luOiAwcHg7XG4gICAgICBwYWRkaW5nLWJvdHRvbTogMTVweDtcbiAgICAgIGZvbnQtc2l6ZTogMC44ZW07XG4gICAgICBmb250LXdlaWdodDogbm9ybWFsO1xuICAgIH1cblxuICAgIC5jbGVhciB7XG4gICAgICBjbGVhcjogYm90aDtcbiAgICB9XG5cbiAgICAuc2Vuc29yX3JpZ2h0IHtcbiAgICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICAgIHJpZ2h0OiAxNXB4O1xuICAgICAgYm90dG9tOiAzOHB4O1xuICAgICAgbWF4LXdpZHRoOiAzMHB4O1xuICAgICAgLnNlbnNvcl91bml0IHtcbiAgICAgICAgbWFyZ2luLXRvcDo4cHg7XG4gICAgICAgIGZvbnQtc2l6ZTogMTRweDtcbiAgICAgIH1cbiAgICAgIC5zZW5zb3JfYXJyb3cge1xuICAgICAgICBtYXJnaW4tdG9wOi0xNXB4O1xuICAgICAgICB3aWR0aDogMTRweDtcbiAgICAgICAgaGVpZ2h0OiA3cHg7XG4gICAgICAgICYuYXJyb3dfdXAgc3ZnIHsgZmlsbDogJGdyZWVuOyB9XG4gICAgICAgICYuYXJyb3dfZG93biBzdmcgeyBmaWxsOiAkcmVkOyB9XG4gICAgICAgICYuZXF1YWwgc3ZnIHsgZmlsbDogJGJsdWVfbGlnaHQ7IH1cbiAgICAgIH1cbiAgICB9XG4gICAgLnNlbnNvcl92YWx1ZSB7XG4gICAgICBtYXJnaW4tdG9wOiAxMHB4O1xuICAgICAgcGFkZGluZy1sZWZ0OiA1MHB4O1xuICAgICAgcGFkZGluZy1yaWdodDogNTBweDtcbiAgICB9XG4gICAgLnNlbnNvcl92YWx1ZV9udWxsIHtcbiAgICAgIG9wYWNpdHk6IDAuMTU7XG4gICAgfVxuICAgIC5zZW5zb3JfaWNvbiB7XG4gICAgICB3aWR0aDogMjJweDtcbiAgICAgIGhlaWdodDogMjJweDtcbiAgICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICAgIGJvdHRvbTogNDVweDtcbiAgICAgIGxlZnQ6IDI1cHg7XG4gICAgICBzdmcge1xuICAgICAgICBmaWxsOiAkZ3JleV9kYXJrZXI7XG4gICAgICB9XG4gICAgfVxuICAgICY6aG92ZXIge1xuICAgICAgb3BhY2l0eTogMC44O1xuICAgIH1cbiAgICAmLnNlbGVjdGVkIHtcbiAgICAgIGNvbG9yOiAkb2ZmX2JsYWNrO1xuICAgICAgLnNlbnNvcl9pY29uIHN2ZyB7XG4gICAgICAgIGZpbGw6ICRvZmZfYmxhY2s7XG4gICAgICB9XG4gICAgfVxuICB9XG59XG5cbi8qKlxuICogIEtJVCBERVRBSUxTXG4gKlxuICovXG5cbiAua2l0X2RldGFpbHNfbm90QXV0aCB7XG4gICAvLyBCdWc6IHRoaXMgd2lsbCBjcmVhdGUgYW4gZXh0cmEgZ2FwIG9uIG1vYmlsZSBkZXZpY2VzXG4gIC8vd2lkdGg6IDUwMHB4O1xuIH1cbi5raXRfZGV0YWlsc19ub3RBdXRoX3RpdGxlIHtcbiAgbWFyZ2luOiAwO1xuIH1cbi5raXRfZGV0YWlsc19ub3RBdXRoX3N1YmhlYWRlciB7XG4gIGNvbG9yOiAkZ3JleV9kYXJrZXI7XG59XG4ua2l0X2RldGFpbHNfbm90QXV0aF9idXR0b24ge1xuICBmbG9hdDogcmlnaHQ7XG59XG4ua2l0X2RldGFpbGVkIHtcbiAgLypoZWlnaHQ6IDEwMDBweDsqL1xuICB6LWluZGV4OiAkbWVkaXVtX3ByaW9yaXR5O1xuICBiYWNrZ3JvdW5kLWNvbG9yOiB3aGl0ZTtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xuXG4gIHAuZGVzY3JpcHRpb24ge1xuICAgIGxpbmUtaGVpZ2h0OiBpbmhlcml0O1xuICAgIGZvbnQtd2VpZ2h0OiBsaWdodGVyO1xuICAgIHdpZHRoOiA3NSU7XG4gIH1cblxuICAua2l0X3RpbWVPcHRze1xuICAgIG1hcmdpbjogMDtcbiAgICBmb250LXNpemU6IDEuM2VtO1xuICB9XG5cbiAgLmtpdF9jaGFydCB7XG4gICAgbWluLWhlaWdodDogMzQwcHg7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogd2hpdGU7XG4gICAgcG9zaXRpb246IHJlbGF0aXZlO1xuXG4gICAgLmhpbnR7XG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkb2ZmX2JsYWNrO1xuICAgICAgb3BhY2l0eTogMC42O1xuICAgIH1cblxuICAgIC5jb250YWluZXIge1xuICAgICAgcGFkZGluZzogMzdweDtcbiAgICAgIGhlaWdodDogMTAwJTtcblxuICAgICAgLmtpdF9jaGFydF9sZWZ0IHtcbiAgICAgICAgY29sb3I6ICRvZmZfYmxhY2s7XG5cbiAgICAgICAgLnNlbnNvcl9zZWxlY3Qge1xuICAgICAgICAgIC5zZW5zb3JfaWNvbl9zZWxlY3RlZCB7XG4gICAgICAgICAgICBmbG9hdDpsZWZ0O1xuICAgICAgICAgICAgbWFyZ2luLXJpZ2h0OiA2cHg7XG4gICAgICAgICAgICBwYWRkaW5nLXRvcDo1cHg7XG4gICAgICAgICAgICBzdmcge1xuICAgICAgICAgICAgICBmaWxsOiAkb2ZmX2JsYWNrO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgICBtZC1zZWxlY3Qge1xuICAgICAgICAgICAgbWFyZ2luOiAwO1xuICAgICAgICAgICAgLm1kLXNlbGVjdC12YWx1ZXtcbiAgICAgICAgICAgICAgLm1kLXRleHR7XG4gICAgICAgICAgICAgICAgLnNlbGVjdF9pbWFnZXtcbiAgICAgICAgICAgICAgICAgIGRpc3BsYXk6bm9uZTtcblxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICAuc2Vuc29yX2Rlc2NyaXB0aW9uIHtcbiAgICAgICAgICAvKi8vY29sb3I6ICRiYWNrZ3JvdW5kX2JsdWVfbGlnaHQ7Ki9cbiAgICAgICAgICAvKi8vaGVpZ2h0OiA4MHB4OyovXG4gICAgICAgICAgLyovL21hcmdpbi10b3A6IDBweDsgLy8zMDsqL1xuICAgICAgICAgIC8qLy9wb3NpdGlvbjogcmVsYXRpdmU7Ki9cblxuICAgICAgICAgIGg2IHtcbiAgICAgICAgICAgIG1hcmdpbjogMDtcbiAgICAgICAgICAgIG1hcmdpbi1ib3R0b206M3B4O1xuICAgICAgICAgICAgZm9udC1zaXplOiAxZW07XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgc21hbGwge1xuICAgICAgICAgICAgbGluZS1oZWlnaHQ6IDFlbTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBhIHtcbiAgICAgICAgICAgIG1hcmdpbi1sZWZ0OiAxMHB4O1xuICAgICAgICAgICAgdGV4dC1kZWNvcmF0aW9uOiBub25lO1xuICAgICAgICAgICAgY29sb3I6ICRncmV5X2RhcmtlcjtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICAuc2Vuc29yX2Rlc2NyaXB0aW9uX2Z1bGwge1xuICAgICAgICAgICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgICAgICAgICAgdG9wOiAtNTBweDtcbiAgICAgICAgICAgIG1hcmdpbi10b3A6IDEwcHg7XG4gICAgICAgICAgICBkaXNwbGF5OiBub25lO1xuICAgICAgICAgICAgb3ZlcmZsb3c6IHZpc2libGU7XG4gICAgICAgICAgICB6LWluZGV4OiAyMDA7XG4gICAgICAgICAgICAvLyBib3JkZXI6IDFweCBzb2xpZCBncmV5O1xuICAgICAgICAgICAgYmFja2dyb3VuZC1jb2xvcjogcmdiYSgyNTUsIDI1NSwgMjU1LCAwLjgpO1xuICAgICAgICAgICAgY29sb3I6ICRiYWNrZ3JvdW5kX2JsdWVfZGFya19jaGFydDtcbiAgICAgICAgICAgIGJvcmRlci1yYWRpdXM6IDJweDtcbiAgICAgICAgICAgIGxpbmUtaGVpZ2h0OiAxLjVlbTtcbiAgICAgICAgICAgIHBhZGRpbmc6IDEwcHg7XG4gICAgICAgICAgICBib3gtc2hhZG93OiAwIDFweCA1cHggcmdiYSgwLCAwLCAwLCAwLjY1KTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICAuc2Vuc29yX2RhdGEge1xuICAgICAgICAgIC5zZW5zb3JfdmFsdWUge1xuICAgICAgICAgICAgZm9udC1zaXplOiAyZW07XG4gICAgICAgICAgICBmb250LXdlaWdodDogYm9sZDtcbiAgICAgICAgICB9XG4gICAgICAgICAgLnNlbnNvcl91bml0IHtcbiAgICAgICAgICAgIGZvbnQtc2l6ZTogLjhlbTtcbiAgICAgICAgICAgIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgICAgICAgICAgIHRvcDogLThweDtcbiAgICAgICAgICB9XG4gICAgICAgICAgLnNlbnNvcl9kYXRhX2ljb24ge1xuICAgICAgICAgICAgd2lkdGg6IDE0cHg7XG4gICAgICAgICAgICBoZWlnaHQ6IDE0cHg7XG4gICAgICAgICAgICBtYXJnaW4tbGVmdDogOHB4O1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICAuc2Vuc29yX2RhdGFfZGVzY3JpcHRpb24ge1xuICAgICAgICAgIGZvbnQtc2l6ZTogMTJweDtcbiAgICAgICAgICBkaXNwbGF5OiBub25lO1xuICAgICAgICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICAgICAgICAtd2Via2l0LXRyYW5zZm9ybTogdHJhbnNsYXRlKDVweCwgNXB4KTtcbiAgICAgICAgICAtbW96LXRyYW5zZm9ybTogdHJhbnNsYXRlKDVweCwgNXB4KTtcbiAgICAgICAgICAtbXMtdHJhbnNmb3JtOiB0cmFuc2xhdGUoNXB4LCA1cHgpO1xuICAgICAgICAgIC1vLXRyYW5zZm9ybTogdHJhbnNsYXRlKDVweCwgNXB4KTtcbiAgICAgICAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZSg1cHgsIDVweCk7XG4gICAgICAgICAgYmFja2dyb3VuZC1jb2xvcjogcmdiYSgyNTUsIDI1NSwgMjU1LCAwLjgpO1xuICAgICAgICAgIGNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2RhcmtfY2hhcnQ7XG4gICAgICAgICAgYm9yZGVyLXJhZGl1czogMnB4O1xuICAgICAgICAgIGxpbmUtaGVpZ2h0OiAxLjVlbTtcbiAgICAgICAgICBwYWRkaW5nOiA0cHggNnB4O1xuICAgICAgICAgIGJveC1zaGFkb3c6IDAgMXB4IDVweCByZ2JhKDAsIDAsIDAsIDAuNjUpO1xuICAgICAgICAgIG1heC13aWR0aDogMTc1cHg7XG4gICAgICAgIH1cbiAgICAgICAgLnNlbnNvcl9jb21wYXJlIHtcbiAgICAgICAgICBmb250LXNpemU6IC44ZW07XG4gICAgICAgICAgcGFkZGluZy10b3A6MzBweDtcbiAgICAgICAgICBtYXJnaW4tdG9wOiAxMHB4O1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC5raXRfY2hhcnRfcmlnaHQge1xuICAgICAgICAvKi8vZGlzcGxheTogaW5saW5lLWJsb2NrOyovXG4gICAgICAgIC8qLy9oZWlnaHQ6IDEwMCU7Ki9cblxuICAgICAgICAuY2hhcnRfbmF2aWdhdGlvbiB7XG4gICAgICAgICAgaGVpZ2h0OiAxMCU7XG4gICAgICAgICAgZm9udC1zaXplOiAxMXB4O1xuICAgICAgICAgID4gZGl2IHtcbiAgICAgICAgICAgIG1hcmdpbi1sZWZ0OiAyMHB4O1xuICAgICAgICAgIH1cbiAgICAgICAgICBpbnB1dCB7XG4gICAgICAgICAgICBtYXJnaW4tbGVmdDogMTBweDtcbiAgICAgICAgICAgIHBhZGRpbmctbGVmdDogMTBweDtcbiAgICAgICAgICAgIGZvbnQtc2l6ZTogMTJweDtcbiAgICAgICAgICB9XG4gICAgICAgICAgLmtpdF90aW1lIHtcbiAgICAgICAgICAgIG1kLWljb24ge1xuICAgICAgICAgICAgICB3aWR0aDogMTFweDtcbiAgICAgICAgICAgICAgaGVpZ2h0OiAxMXB4O1xuICAgICAgICAgICAgICBtYXJnaW4tcmlnaHQ6IDhweDtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG5cbiAgICAgICAgICAucGlja2VyX2NvbnRhaW5lciB7XG4gICAgICAgICAgICBtYXJnaW4tbGVmdDogMjBweDtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBAbWVkaWEgb25seSBzY3JlZW4gYW5kIChtYXgtZGV2aWNlLXdpZHRoOiAxMDUwcHgpIHtcbiAgICAgICAgICAgIC5jaGFydF9uYXZpZ2F0aW9ue1xuICAgICAgICAgICAgICBqdXN0aWZ5LWNvbnRlbnQ6IHNwYWNlLWFyb3VuZCAhaW1wb3J0YW50O1xuICAgICAgICAgICAgICBtYXJnaW46IDVweCAhaW1wb3J0YW50O1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgLnBpY2tlcl9jb250YWluZXIge1xuICAgICAgICAgICAgICBtYXJnaW46IDAgNXB4IDAgNXB4ICFpbXBvcnRhbnQ7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgQG1lZGlhIG9ubHkgc2NyZWVuIGFuZCAobWF4LWRldmljZS13aWR0aDogOTYwcHgpIHtcblxuICAgICAgICAgICAgLnBpY2tlcl9jb250YWluZXIge1xuICAgICAgICAgICAgICBkaXNwbGF5OiBmbGV4ICFpbXBvcnRhbnQ7XG4gICAgICAgICAgICAgIGFsaWduLWNvbnRlbnQ6IG1pZGRsZSAhaW1wb3J0YW50O1xuICAgICAgICAgICAgICBtYXJnaW46IDEwcHggMCAwIDAgIWltcG9ydGFudDtcbiAgICAgICAgICAgICAgd2lkdGg6IDEwMCUgIWltcG9ydGFudDtcbiAgICAgICAgICAgICAganVzdGlmeS1jb250ZW50OiBzcGFjZS1iZXR3ZWVuICFpbXBvcnRhbnQ7XG4gICAgICAgICAgICAgIGFsaWduLWl0ZW1zOiBjZW50ZXIgIWltcG9ydGFudDtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgLndvcmRfcGlja2Vye1xuICAgICAgICAgICAgICBkaXNwbGF5OiBpbmxpbmUtYmxvY2sgIWltcG9ydGFudDtcbiAgICAgICAgICAgICAgd2lkdGg6IDEwMCUgIWltcG9ydGFudDtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIC5jaGFydF9tb3ZlIHtcbiAgICAgICAgICAgICAgZGlzcGxheTogZmxleCAhaW1wb3J0YW50O1xuICAgICAgICAgICAgICB3aWR0aDogMTAwJSAhaW1wb3J0YW50O1xuICAgICAgICAgICAgICBqdXN0aWZ5LWNvbnRlbnQ6IHNwYWNlLWJldHdlZW4gIWltcG9ydGFudDtcbiAgICAgICAgICAgICAgbWFyZ2luLXRvcDogMTVweCAhaW1wb3J0YW50O1xuXG4gICAgICAgICAgICAgIC5jaGFydF9tb3ZlX2J1dHRvbiB7XG4gICAgICAgICAgICAgICAgd2lkdGg6IDEwMCUgIWltcG9ydGFudDtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC5jaGFydF9tb3ZlIHtcbiAgICAgICAgICAgIC5jaGFydF9tb3ZlX2J1dHRvbiB7XG4gICAgICAgICAgICAgIG1hcmdpbjogMHB4IC0ycHggMCAtMnB4O1xuICAgICAgICAgICAgICB3aWR0aDogNDBweDtcbiAgICAgICAgICAgICAgaGVpZ2h0OiAyOHB4O1xuICAgICAgICAgICAgICBkaXNwbGF5OiBpbmxpbmUtZmxleDtcbiAgICAgICAgICAgICAgYm9yZGVyOiAxcHggc29saWQgJGdyZXlfZGFya2VyO1xuICAgICAgICAgICAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkZ3JleV9saWdodGVzdDtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIC5jaGFydF9tb3ZlX3JpZ2h0IHtcbiAgICAgICAgICAgICAgLy8gbWFyZ2luLWxlZnQ6IDRweDtcbiAgICAgICAgICAgICAgYm9yZGVyLXRvcC1sZWZ0LXJhZGl1czogMHB4O1xuICAgICAgICAgICAgICBib3JkZXItYm90dG9tLWxlZnQtcmFkaXVzOiAwcHg7XG4gICAgICAgICAgICAgIHN2ZyB7XG4gICAgICAgICAgICAgICAgZmlsbDogJGdyZXlfZGFya2VyO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICY6aG92ZXIge1xuICAgICAgICAgICAgICAgIGJhY2tncm91bmQtY29sb3I6ICRncmV5X2RhcmtlcjtcbiAgICAgICAgICAgICAgICBzdmcge1xuICAgICAgICAgICAgICAgICAgZmlsbDogJGdyZXlfbGlnaHRlc3Q7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICAuY2hhcnRfbW92ZV9sZWZ0IHtcbiAgICAgICAgICAgICAgc3ZnIHtcbiAgICAgICAgICAgICAgICBmaWxsOiAkZ3JleV9kYXJrZXI7XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgJjpob3ZlciB7XG4gICAgICAgICAgICAgICAgYmFja2dyb3VuZC1jb2xvcjogJGdyZXlfZGFya2VyO1xuICAgICAgICAgICAgICAgIHN2ZyB7XG4gICAgICAgICAgICAgICAgICBmaWxsOiAkZ3JleV9saWdodGVzdDtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIG1kLWljb24ge1xuICAgICAgICAgICAgICB3aWR0aDogNS41cHg7XG4gICAgICAgICAgICAgIGhlaWdodDogOXB4O1xuICAgICAgICAgICAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gICAgICAgICAgICAgIHRvcDogMHB4O1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgfVxuICAua2l0X2RldGFpbHMge1xuICAgIGNvbG9yOiAkYmFja2dyb3VuZF9ibHVlX2RhcmtfbmF2YmFyO1xuICB9XG4gIC5raXRfZGV0YWlsc19jb250ZW50IHtcbiAgICBzdmcge1xuICAgICAgZmlsbDogJG9mZl9ibGFjaztcbiAgICB9XG5cbiAgICAua2l0X2RldGFpbHNfbGFiZWxze1xuICAgICAgbWFyZ2luLXRvcDozMHB4O1xuICAgIH1cbiAgfVxuICAua2l0X2RldGFpbHNfbmFtZSB7XG4gICAgbWFyZ2luOiA1cHggMCAyMHB4IDA7XG4gIH1cbiAgLmtpdF9kZXRhaWxzX3R5cGUge1xuICAgID4gc3BhbiB7XG4gICAgICBtYXJnaW4tcmlnaHQ6IDE1cHg7XG4gICAgfVxuICB9XG4gIC5raXRfZGV0YWlsc190eXBlX2xvY2F0aW9uIHN2ZyB7XG4gICAgZmlsbDogJHNlY29uZGFyeV9jb2xvcjtcbiAgfVxuICAua2l0X2RldGFpbHNfdmVyc2lvbiB7XG4gICAgc3BhbiB7XG4gICAgICBjb2xvcjogJHNlY29uZGFyeV9jb2xvcl9saWdodDtcbiAgICAgIGZvbnQtd2VpZ2h0OiBib2xkO1xuICAgIH1cbiAgfVxuICAua2l0X2RldGFpbHNfaWQge1xuICAgIHNwYW4ge1xuICAgICAgY29sb3I6ICRzZWNvbmRhcnlfY29sb3JfbGlnaHQ7XG4gICAgICBmb250LXdlaWdodDogYm9sZDtcbiAgICB9XG4gIH1cblxuICAua2l0X2RldGFpbHNfZGVzY3JpcHRpb24ge1xuICAgIHBhZGRpbmctdG9wOiA0cHg7XG4gIH1cbiAgLmtpdF9kZXRhaWxzX2Rlc2NyaXB0aW9uX3RpdGxlIHtcbiAgICBmb250LXNpemU6IDE2cHg7XG4gICAgbWFyZ2luLWJvdHRvbTogMTVweDtcbiAgfVxuICAua2l0X2RldGFpbHNfZGVzY3JpcHRpb25fY29udGVudCB7XG4gICAgZm9udC1zaXplOiAxMnB4O1xuICB9XG4gIC5raXRfZGV0YWlsc19tYW5hZ2VfYnV0dG9ucyB7XG4gICAgbWFyZ2luLWJvdHRvbTogMzJweDtcbiAgfVxuICAua2l0X2RldGFpbHNfYnV0dG9uX2RlbGV0ZSB7XG4gICAgY29sb3I6ICRyZWQ7XG4gIH1cbiAgLmtpdF9kZXRhaWxzX21hbmFnZV9kYXRhIHtcbiAgICAuZGF0YV9jb250YWluZXIge1xuICAgIH1cbiAgfVxuICAua2l0X2RldGFpbHNfa2V5IHtcbiAgICBmbG9hdDogbGVmdDtcbiAgICBoZWlnaHQ6IDI2cHg7XG4gIH1cbiAgLm1kLWJ1dHRvbi5raXRfZGV0YWlsc19rZXlfcmVmcmVzaEJ1dHRvbjpob3ZlciwgLm1kLWJ1dHRvbi5raXRfZGV0YWlsc19rZXlfcmVmcmVzaEJ1dHRvbjpmb2N1cyB7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogI0RCREJEQjtcbiAgfVxuICAua2l0X2RldGFpbGVkX2ljb25fY29udGVudCB7XG4gICAgd2lkdGg6IDE0cHg7XG4gICAgaGVpZ2h0OiAxNHB4O1xuICAgIG1hcmdpbi1yaWdodDogNHB4O1xuICAgIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgICB0b3A6IC0xcHg7XG4gICAgc3ZnIHtcbiAgICAgIGZpbGw6ICRncmV5X2RhcmtlcjtcbiAgICB9XG4gIH1cbiAgLmtpdF9kZXRhaWxlZF9pY29uX3RpdGxlIHtcbiAgICB3aWR0aDogMThweDtcbiAgICBoZWlnaHQ6IDE4cHg7XG4gICAgbWFyZ2luLXJpZ2h0OiA4cHg7XG4gICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgIHRvcDogLTJweDtcbiAgfVxuICAua2l0X2RldGFpbGVkX3RpdGxlX2NvbnRhaW5lciB7XG4gICAgcGFkZGluZzogMjBweCAwO1xuICB9XG4gIC5raXRfZGV0YWlsZWRfdGl0bGUge1xuICAgIGZvbnQtc2l6ZTogMTZweDtcbiAgICBkaXNwbGF5OiBpbmxpbmU7XG4gIH1cbiAgLmtpdF9kZXRhaWxlZF9jb250ZW50X2NvbnRhaW5lciB7XG4gICAgbWFyZ2luOiA2MnB4IDM3cHggMHB4O1xuXG4gICAgLmtpdF9kZXRhaWxzX2NvbnRlbnRfbWFpbiB7XG4gICAvLyBCdWc6IHRoaXMgd2lsbCBjcmVhdGUgYW4gZXh0cmEgZ2FwIG9uIG1vYmlsZSBkZXZpY2VzXG4gICAgLy8gIHdpZHRoOiA1MDBweDtcbiAgICB9XG4gIH1cbiAgLmtpdF9vd25lciB7XG4gICAgY29sb3I6ICRiYWNrZ3JvdW5kX2JsdWVfZGFya19uYXZiYXI7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogI0Y1RjVGNTtcbiAgICBwYWRkaW5nLWJvdHRvbTo0MHB4O1xuXG4gICAgLmNvbnRhaW5lciB7XG4gICAgICB3aWR0aDogOTUlO1xuICAgICAgbWFyZ2luOiAwIGF1dG87XG4gICAgfVxuICAgIC5raXRfb3duZXJfdXNlcm5hbWVMaW5rIHtcbiAgICAgIHRleHQtZGVjb3JhdGlvbjogbm9uZTtcbiAgICB9XG4gICAgLmtpdF9vd25lcl91c2VybmFtZVRleHQge1xuICAgICAgY29sb3I6ICRvZmZfYmxhY2s7XG4gICAgICBtYXJnaW46IDUwcHggMCAyMHB4IDA7XG4gICAgfVxuICAgIC5raXRfb3duZXJfbG9jYXRpb24ge1xuICAgICAgY29sb3I6ICRzZWNvbmRhcnlfY29sb3I7XG4gICAgICBmb250LXdlaWdodDogYm9sZDtcbiAgICB9XG4gICAgLmtpdF9vd25lcl91cmwgYSB7XG4gICAgICB0ZXh0LWRlY29yYXRpb246IG5vbmU7XG4gICAgICBjb2xvcjogJGJhY2tncm91bmRfYmx1ZV9kYXJrX25hdmJhcjtcbiAgICB9XG4gICAgLmtpdF9vd25lcl9raXRzX3RpdGxlIHtcbiAgICAgIG1hcmdpbi1ib3R0b206IDUwcHg7XG4gICAgfVxuICAgIC5raXRfb3duZXJfa2l0c19saXN0IHtcbiAgICAgIGJvcmRlcjogMXB4IHNvbGlkICNFQUVDRjE7XG4gICAgICBoZWlnaHQ6IDgycHg7XG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yOiB3aGl0ZTtcbiAgICB9XG4gICAgLmtpdF9vd25lcl9raXRzX2xpc3RfYXZhdGFyIHtcbiAgICAgIGZsb2F0OiBsZWZ0O1xuICAgICAgd2lkdGg6IDQ0cHg7XG4gICAgICBoZWlnaHQ6IDQ0cHg7XG4gICAgICBtYXJnaW4tcmlnaHQ6IDMlO1xuICAgICAgbWFyZ2luLXRvcDogOXB4O1xuICAgIH1cbiAgICAua2l0X293bmVyX2tpdHNfbGlzdF9jb250ZW50IHtcbiAgICAgIGZsb2F0OiBsZWZ0O1xuICAgICAgbWFyZ2luLXRvcDogNXB4O1xuICAgIH1cbiAgICAua2l0X293bmVyX2tpdHNfbGlzdF9kYXRhIHtcbiAgICAgIGZvbnQtc2l6ZTogMTZweDtcbiAgICAgIHNwYW4ge1xuICAgICAgICBtYXJnaW4tcmlnaHQ6IDE1cHg7XG4gICAgICB9XG4gICAgfVxuICAgIC5raXRfb3duZXJfa2l0c19saXN0X3N0YXRlIHtcbiAgICAgIGRpc3BsYXk6IGlubGluZTtcbiAgICAgIGZvbnQtc2l6ZTogMTJweDtcbiAgICAgIGJvcmRlci1yYWRpdXM6IDIwcHg7XG4gICAgICBwYWRkaW5nOiA2cHggMTFweDtcbiAgICAgIG1hcmdpbjogMTVweDtcbiAgICB9XG4gICAgLmtpdF9vd25lcl9raXRzX2xpc3RfbmV2ZXJfcHVibGlzaGVkIHtcbiAgICAgIGJhY2tncm91bmQtY29sb3I6ICR5ZWxsb3c7XG4gICAgfVxuICAgIC5raXRfb3duZXJfa2l0c19saXN0X25vdF9jb25maWd1cmVkIHtcbiAgICAgIGJhY2tncm91bmQtY29sb3I6ICRyZWQ7XG4gICAgfVxuICAgIC5raXRfb3duZXJfa2l0c19saXN0X2NvbmZpZyB7XG4gICAgICBmbG9hdDogcmlnaHQ7XG4gICAgICBtYXJnaW4tdG9wOiAxM3B4O1xuICAgIH1cbiAgICAua2l0X293bmVyX2tpdHNfbGlzdF9jb25maWdfaWNvbiB7XG4gICAgICB3aWR0aDogMjJweDtcbiAgICAgIGhlaWdodDogMjJweDtcbiAgICB9XG4gIH1cbn1cblxuLnNlbnNvcl92YWx1ZSB7XG4gIGZvbnQtc2l6ZTogMzRweDtcbiAgZm9udC1mYW1pbHk6IEthbml0O1xuICBmb250LXdlaWdodDogNzAwO1xufVxuXG4vKiovXG5cbi8qIFshXSBMZWF2ZSBhbmltYXRpb24gKG5nLWxlYXZlICkgYXJlIGRpc2FibGVkXG5iZWNhdXNlIHRoZXkgY29uZmxpY3Qgd2l0aCBlbnRlciBhbmltYXRpb25zICovXG5cblxuLnNlbnNvcl9hbmltYXRpb24ubmctZW50ZXIge1xuICB0cmFuc2l0aW9uOjFzIGVhc2Utb3V0IGFsbDtcbn1cbi5zZW5zb3JfYW5pbWF0aW9uLm5nLWVudGVyIHtcbiAgdHJhbnNmb3JtOiBzY2FsZSgxLjA1KTtcbiAgLy8gb3BhY2l0eTogMDtcbn1cbi5zZW5zb3JfYW5pbWF0aW9uLm5nLWVudGVyLWFjdGl2ZSB7XG4gIHRyYW5zZm9ybTogc2NhbGUoMSk7XG4gIC8vIG9wYWNpdHk6IDE7XG59XG5cbi8qIC5zZW5zb3JfYW5pbWF0aW9uLm5nLWxlYXZle1xuICB0cmFuc2l0aW9uOjFzIGVhc2UgYWxsO1xufVxuLnNlbnNvcl9hbmltYXRpb24ubmctbGVhdmV7XG4gIG9wYWNpdHk6IDA7XG59XG4uc2Vuc29yX2FuaW1hdGlvbi5uZy1sZWF2ZS1hY3RpdmV7XG4gIG9wYWNpdHk6IDE7XG59ICovXG5cbi8qIFRpbWUgYW5pbWF0aW9uIGN1cnJlbnRseSBub3QgaW4gdXNlICovXG5cbi50aW1lX2FuaW1hdGlvbi5uZy1lbnRlciB7XG4gIHRyYW5zaXRpb246MXMgZWFzZS1vdXQgYWxsO1xufVxuLnRpbWVfYW5pbWF0aW9uLm5nLWVudGVyIHtcbiAgdHJhbnNmb3JtOiBzY2FsZSgxLjA1KTtcbn1cbi50aW1lX2FuaW1hdGlvbi5uZy1lbnRlci1hY3RpdmUge1xuICB0cmFuc2Zvcm06IHNjYWxlKDEpO1xufVxuXG4vKiAudGltZV9hbmltYXRpb24ubmctbGVhdmUge1xuICB0cmFuc2l0aW9uOjFzIGVhc2UgYWxsO1xufVxuLnRpbWVfYW5pbWF0aW9uLm5nLWxlYXZlIHtcbiAgb3BhY2l0eTogMDtcbn1cbi50aW1lX2FuaW1hdGlvbi5uZy1sZWF2ZS1hY3RpdmUge1xuICBvcGFjaXR5OiAxO1xufSAqL1xuXG4uYmF0X2FuaW1hdGlvbi5uZy1lbnRlciB7XG4gIHRyYW5zaXRpb246MXMgZWFzZS1vdXQgYWxsO1xufVxuLmJhdF9hbmltYXRpb24ubmctZW50ZXIge1xuICB0cmFuc2Zvcm06IHNjYWxlKDEuMDUpO1xufVxuLmJhdF9hbmltYXRpb24ubmctZW50ZXItYWN0aXZlIHtcbiAgdHJhbnNmb3JtOiBzY2FsZSgxKTtcbn1cblxuLyogLmJhdF9hbmltYXRpb24ubmctbGVhdmUge1xuICB0cmFuc2l0aW9uOjJzIGVhc2UgYWxsO1xufVxuLmJhdF9hbmltYXRpb24ubmctbGVhdmUge1xuICBvcGFjaXR5OiAwO1xufVxuLmJhdF9hbmltYXRpb24ubmctbGVhdmUtYWN0aXZlIHtcbiAgb3BhY2l0eTogMTtcbn0gKi9cbiIsIi8qXG4gIE1PREFMIFNUT1JFXG4qL1xuXG4uc3RvcmVfaXRlbURlc2NyaXB0aW9uIHtcbiAgbWFyZ2luLWxlZnQ6MjBweDtcbiAgdWwge1xuICAgIGxpc3Qtc3R5bGUtaW1hZ2U6IHVybCgnLi4vYXNzZXRzL2ltYWdlcy9jaGVja19jaXJjbGUuc3ZnJyk7XG4gICAgbGlzdC1zdHlsZS1wb3NpdGlvbjogaW5oZXJpdDtcbiAgICBmb250LXNpemU6IC44ZW07XG4gICAgbGluZS1oZWlnaHQ6IDEuOGVtO1xuICB9XG59XG4iLCIuY3N2X2ZpbGVfaXRlbSB7XG4gIGRpc3BsYXk6IGZsZXg7XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIG1kLWNoZWNrYm94IHtcbiAgICBtYXJnaW4tYm90dG9tOiAwcHg7XG4gIH1cbiAgc3BhbiB7XG4gICAgbWFyZ2luLXJpZ2h0OiAyMHB4XG4gIH1cbn1cblxuLmxpc3Qtc2hhZG93IHtcbiAgYm94LXNoYWRvdzogMHB4IDFweCAzcHggMHB4IHJnYmEoMCwwLDAsMC4yKTtcbiAgbWQtbGlzdC1pdGVtIHtcbiAgICBib3JkZXItYm90dG9tOiBzb2xpZCAxcHggJGdyZXlfbGlnaHRlcjtcbiAgfVxuICBtZC1saXN0LWl0ZW06bGFzdC1jaGlsZCB7XG4gICAgYm9yZGVyLWJvdHRvbTogbm9uZTtcbiAgfVxufVxuXG4vLyAubWQtYmFyLmJnLWdyZWVuIHtcbi8vICAgYmFja2dyb3VuZC1jb2xvcjogJGdyZWVuO1xuLy8gfVxuXG4ubGFiZWwtZ3JleSB7XG4gIGJhY2tncm91bmQtY29sb3I6ICRncmV5O1xufVxuXG4udGl0bGUtdGltZWxpbmUge1xuICBmb250LXNpemU6IDMwcHg7XG59XG5cbi8qIFRoaXMgaXMgdGVtcG9yYXJ5ICovXG5cbi51cGxvYWQtY3N2IHtcbiAgICAuY29udGFpbmVyIHtcbiAgICAgIC8vIHBhZGRpbmctYm90dG9tOjIwcHg7XG4gICAgICBtYXJnaW4tcmlnaHQ6IGF1dG87XG4gICAgICBtYXJnaW4tbGVmdDogYXV0bztcbiAgICB9XG5cbiAgICAuY3N2X2NvbnRlbnR7XG4gICAgICBtYXJnaW4tdG9wOiAyNDBweDtcbiAgICAgIG1pbi1oZWlnaHQ6IDUwdmg7XG4gICAgfVxuXG4gICAgQG1lZGlhIChtaW4td2lkdGg6IDc2OHB4KSB7XG4gICAgICAgIC5jb250YWluZXIge1xuICAgICAgICAgICAgd2lkdGg6IDc1MHB4O1xuICAgICAgICB9XG4gICAgfVxuXG4gICAgQG1lZGlhIChtaW4td2lkdGg6IDk5MnB4KSB7XG4gICAgICAgIC5jb250YWluZXIge1xuICAgICAgICAgICAgd2lkdGg6IDk3MHB4O1xuICAgICAgICB9XG4gICAgfVxuXG4gICAgQG1lZGlhIChtaW4td2lkdGg6IDEyMDBweCkge1xuICAgICAgICAuY29udGFpbmVyIHtcbiAgICAgICAgICAgIHdpZHRoOiAxMTcwcHg7XG4gICAgICAgIH1cbiAgICB9XG59XG4iLCIvKipcbiAqICBCQUNLRFJPUCBDT01QT05FTlRcbiAqL1xuXG4vLyBQb3B1cCBiYWNrZHJvcFxuXG5tZC1jb250ZW50Lm1kLW9wYXF1ZSB7XG4gIGJhY2tncm91bmQtY29sb3I6IHJnYmEoMzAsIDMwLCAzMCwgMC42KTtcbn1cblxuLy9Mb2FkaW5nIGJhY2tkcm9wXG5cbm1kLWNvbnRlbnQubWQtbWFpbkJhY2tkcm9wIHtcbiAgZGlzcGxheTogZmxleDtcbiAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAganVzdGlmeS1jb250ZW50OiBjZW50ZXI7XG4gIHdpZHRoOiAxMDB2dztcbiAgaGVpZ2h0OiAxMDB2aDtcbiAgLmJhY2tkcm9wX2ljb24ge1xuICAgIHdpZHRoOiAxMDFweDtcbiAgICBoZWlnaHQ6IDEwMXB4O1xuICB9XG59XG5cbi8vIHN0YXRlIGNoYW5nZSBiYWNrZHJvcFxubWQtY29udGVudC5tZC1zdGF0ZUNoYW5nZUJhY2tkcm9wIHtcbiAgei1pbmRleDogOTk5O1xuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIGJhY2tncm91bmQtY29sb3I6IHRyYW5zcGFyZW50O1xuICB3aWR0aDogMTAwdnc7XG4gIGhlaWdodDogMTAwdmg7XG59XG5cbi5iYWNrZHJvcF9pY29uIHtcbiAgc3ZnIHtcbiAgICBvdmVyZmxvdzogdmlzaWJsZTtcbiAgICBmaWxsOiAkZ3JleV9kYXJrZXI7XG4gIH1cbiAgLnNjLWNpcmNsZS1hcnJvdyB7XG4gICAgLXdlYmtpdC10cmFuc2Zvcm0tb3JpZ2luOiA1MCUgNTMuOCU7XG4gICAgLW1vei10cmFuc2Zvcm0tb3JpZ2luOiA1MCUgNTMuOCU7XG4gICAgLW8tdHJhbnNmb3JtLW9yaWdpbjogNTAlIDUzLjglO1xuICAgIHRyYW5zZm9ybS1vcmlnaW46IDUwJSA1My44JTtcbiAgICAtd2Via2l0LWFuaW1hdGlvbjpzcGluIDNzIGxpbmVhciBpbmZpbml0ZTtcbiAgICAtbW96LWFuaW1hdGlvbjpzcGluIDNzIGxpbmVhciBpbmZpbml0ZTtcbiAgICBhbmltYXRpb246c3BpbiAzcyBsaW5lYXIgaW5maW5pdGU7XG4gIH1cbn1cblxuQC1tb3ota2V5ZnJhbWVzIHNwaW4geyAxMDAlIHsgLW1vei10cmFuc2Zvcm06IHJvdGF0ZSgzNjBkZWcpOyB9IH1cbkAtd2Via2l0LWtleWZyYW1lcyBzcGluIHsgMTAwJSB7IC13ZWJraXQtdHJhbnNmb3JtOiByb3RhdGUoMzYwZGVnKTsgfSB9XG5Aa2V5ZnJhbWVzIHNwaW4geyAxMDAlIHsgLXdlYmtpdC10cmFuc2Zvcm06IHJvdGF0ZSgzNjBkZWcpOyB0cmFuc2Zvcm06cm90YXRlKDM2MGRlZyk7IH0gfVxuIiwic2VjdGlvbiAuc3RpY2tNZW51IHtcbiAgcG9zaXRpb246IGFic29sdXRlICFpbXBvcnRhbnQ7XG4vLyAgdG9wOiA2NHB4O1xufVxuXG4uc3RpY2tOYXYge1xuICBwb3NpdGlvbjogZml4ZWQ7XG4gIHRvcDogMHB4O1xuICBib3JkZXItYm90dG9tOjFweCBzb2xpZCAjNjY2O1xufVxuIl19 */