diff --git a/exercises/ardrone_navigation/CMakeLists.txt b/exercises/ardrone_navigation/CMakeLists.txt new file mode 100644 index 000000000..8c27b9c9d --- /dev/null +++ b/exercises/ardrone_navigation/CMakeLists.txt @@ -0,0 +1,14 @@ + +if( ICE_PY_MODULES AND NUMPY_MODULES AND OpenCV_PY_MODULES AND QT4_PY_MODULES AND QWT5_PY_MODULES) + #install + #INSTALL (DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/ DESTINATION share/jderobot/python/introrob_py + # COMPONENT core + # FILES_MATCHING + # PATTERN "*.py" + #) + + #INSTALL (PROGRAMS ${CMAKE_CURRENT_SOURCE_DIR}/introrob_py DESTINATION bin OPTIONAL COMPONENT core) + +endif(ICE_PY_MODULES AND NUMPY_MODULES AND OpenCV_PY_MODULES AND QT4_PY_MODULES AND QWT5_PY_MODULES) + + diff --git a/exercises/ardrone_navigation/MyAlgorithm.py b/exercises/ardrone_navigation/MyAlgorithm.py new file mode 100644 index 000000000..2e87bcbb2 --- /dev/null +++ b/exercises/ardrone_navigation/MyAlgorithm.py @@ -0,0 +1,103 @@ +import threading +import time +from datetime import datetime + +import math +import jderobot + +from parallelIce.cameraClient import CameraClient +from parallelIce.navDataClient import NavDataClient +from parallelIce.cmdvel import CMDVel +from parallelIce.extra import Extra +from parallelIce.pose3dClient import Pose3DClient + +from navigation.navigation import navigation +from navigation.control.ardroneControl import ardroneControl + +time_cycle = 80 + +class MyAlgorithm(threading.Thread): + + def __init__(self, camera, navdata, pose, cmdvel, extra): + self.camera = camera + self.navdata = navdata + self.pose = pose + self.cmdvel = cmdvel + self.extra = extra + + self.minError=0.01 + + self.stop_event = threading.Event() + self.kill_event = threading.Event() + self.lock = threading.Lock() + threading.Thread.__init__(self, args=self.stop_event) + + self.navigation = navigation(None,[0,0,0],0.2,3) + + self.find_path = False + self.get_point = True + #self.goal_point = [[7.5,7.5,1],[-5,-7.5,1],[-7.5,-7.5,5],[-7.5,7.5,5],[7.5,-7.5,5]] + self.goal_point = [[7.5, 0 ,6], [7.5,7.5,2],[-7.5, 7.5, 2],[-7.5,7.5,6]] + self.num = 0 + + def run (self): + + self.stop_event.clear() + + while (not self.kill_event.is_set()): + + start_time = datetime.now() + + if not self.stop_event.is_set(): + self.execute() + + finish_Time = datetime.now() + + dt = finish_Time - start_time + ms = (dt.days * 24 * 60 * 60 + dt.seconds) * 1000 + dt.microseconds / 1000.0 + #print (ms) + if (ms < time_cycle): + time.sleep((time_cycle - ms) / 1000.0) + + def stop (self): + self.stop_event.set() + + def play (self): + if self.is_alive(): + self.stop_event.clear() + else: + self.start() + + def kill (self): + self.kill_event.set() + + + def execute(self): + # Add your code here + # use: self.pose.getPose3d().x + # self.pose.getPose3d().y + # to get the coordinates of the drone in the x,y plane + pose_n = [self.pose.getPose3d().x, self.pose.getPose3d().y, self.pose.getPose3d().z, self.pose.getPose3d().yaw] + print (pose_n) + + if not self.find_path and self.get_point and self.num < len(self.goal_point): + self.cmdvel.setVX(0) + self.cmdvel.setVY(0) + self.cmdvel.setVZ(0) + self.cmdvel.setYaw(0) + self.cmdvel.sendVelocities() + if self.num == 0: + self.navigation.update_map_3d() + goal_pose = self.goal_point[self.num] + print ("goal", goal_pose) + self.find_path = self.navigation.path_planning_3d(pose_n,goal_pose) + else: + [self.get_point, data] = self.navigation.path_following_3d(pose_n) + if self.get_point: + self.find_path = False + self.num += 1 + self.cmdvel.setVX(data[0]) + self.cmdvel.setVY(data[1]) + self.cmdvel.setVZ(data[2]) + self.cmdvel.setYaw(data[3]) + self.cmdvel.sendVelocities() diff --git a/exercises/ardrone_navigation/Navigation3dOmpl.py b/exercises/ardrone_navigation/Navigation3dOmpl.py new file mode 100644 index 000000000..c4f6fb1dd --- /dev/null +++ b/exercises/ardrone_navigation/Navigation3dOmpl.py @@ -0,0 +1,72 @@ +#!/usr/bin/python3 +# +# Copyright (C) 1997-2016 JDE Developers Team +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# Authors : +# Alberto Martin Florido +# Aitor Martinez Fernandez +# + +import sys +import config +import comm +from MyAlgorithm import MyAlgorithm +import easyiceconfig as EasyIce +from gui.threadGUI import ThreadGUI +from parallelIce.cameraClient import CameraClient +from parallelIce.navDataClient import NavDataClient +from parallelIce.cmdvel import CMDVel +from parallelIce.extra import Extra +from parallelIce.pose3dClient import Pose3DClient +from gui.GUI import MainWindow +from PyQt5.QtWidgets import QApplication + +import signal + +signal.signal(signal.SIGINT, signal.SIG_DFL) + +if __name__ == '__main__': + + cfg = config.load(sys.argv[1]) + + #starting comm + jdrc= comm.init(cfg, 'Introrob') + + camera = jdrc.getCameraClient("Introrob.Camera") + navdata = jdrc.getNavdataClient("Introrob.Navdata") + pose = jdrc.getPose3dClient("Introrob.Pose3D") + cmdvel = jdrc.getCMDVelClient("Introrob.CMDVel") + extra = jdrc.getArDroneExtraClient("Introrob.Extra") + + + algorithm=MyAlgorithm(camera, navdata, pose, cmdvel, extra) + + app = QApplication(sys.argv) + frame = MainWindow() + frame.setCamera(camera) + frame.setNavData(navdata) + frame.setPose3D(pose) + frame.setCMDVel(cmdvel) + frame.setExtra(extra) + frame.setAlgorithm(algorithm) + frame.show() + + + + t2 = ThreadGUI(frame) + t2.daemon=True + t2.start() + + sys.exit(app.exec_()) diff --git a/exercises/ardrone_navigation/README.md b/exercises/ardrone_navigation/README.md new file mode 100644 index 000000000..589f82470 --- /dev/null +++ b/exercises/ardrone_navigation/README.md @@ -0,0 +1,70 @@ +# Navigation3DOmpl Exercise + +In this practice we will learn the use of ompl to implement a local navigation algorithm in the quadricopters. +For this practice a world has been designed for the Gazebo simulator. This world has a 3D model of the AR.Drone and a indoor environment. + +You can see this [repo](https://github.com/TheRoboticsClub/colab-gsoc2018-HanqingXie) and [web](https://jderobot.org/Club-hanqingxie) for more detials. + +## Dependency +This project requires the follow dependency: +* jedrobot +* python2.7 +* [OMPL](http://ompl.kavrakilab.org/) + +### install ompl +[Download the OMPL installation script](http://ompl.kavrakilab.org/install-ompl-ubuntu.sh). First, make the script executable: + +```chmod u+x install-ompl-ubuntu.sh``` + +Next, there are three ways to run this script: + +```./install-ompl-ubuntu.sh --python will install OMPL with Python bindings``` + +## How to run + +1. Copy the inHouse model into jderobot gazebo models + + `$ cp -r models/inHouse /root_jderobot/share/jderobot/gazebo/models/` + +2. In a terminal launch the gazebo simulator: + + `$ gazebo ardrone-inHouse.world` + +3. In other terminal launch the position_control component with your algorithm: + + `$ python2 ./Navigation3dOmpl.py ardrone_conf.yml` + +## How to do the practice +To carry out the practice, you must edit the MyAlgorithm.py file and insert the control logic into it. + +## Where to insert the code +[MyAlgorithm.py](MyAlgorithm.py#L65) +``` + def execute(self): + + # Add your code here + print "Runing" + + #EXAMPLE OF HOW TO SEND INFORMATION TO THE ROBOT ACTUATORS + #self.motors.sendV(10) + #self.motors.sendW(5) + # TODO + +``` + +## API + +* self.pose.getPose3d().x - returns the position values ​​of the drone in X axis +* self.pose.getPose3d().y - returns the position values ​​of the drone in Y axis +* self.pose.getPose3d().roll, self.pose.getPose3d().pitch, self.pose.getPose3d().yaw: returns the rotation values ​​of the drone in space. +* self.camera.getImage().data: returns the image captured by the active camera of the drone (frontal or ventral). +* self.cmdvel.setVX(velx): set linear speed of the drone. +* self.cmdvel.setVY(vely): set linear speed of the drone. +* self.cmdvel.setVZ(vely): set linear speed of the drone. +* self.cmdvel.setVYaw(vely): set angular speed of the drone. +* self.cmdvel.sendVelocities(): send set velocities to the drone. +* self.cmdvel.sendCMDVel(self,vx,vy,vz,ax,ay,az): sends linear and angular speed commands to the drone. +* self.extra.toggleCam(): changes the drone's active camera (frontal or the one below). +* self.extra.takeOff(): Takeoff of the drone. +* self.extra.land(): landing of the drone. + diff --git a/exercises/ardrone_navigation/ardrone-inHouse.world b/exercises/ardrone_navigation/ardrone-inHouse.world new file mode 100644 index 000000000..c60c15692 --- /dev/null +++ b/exercises/ardrone_navigation/ardrone-inHouse.world @@ -0,0 +1,29 @@ + + + + + + model://ground_plane + + + + model://ArDrone2 + 7.5 0 4.3 0 0 0 + + + + model://sun + 0 0 3 0 0 0 + + + model://sun + + + + inHouse + model://inHouse + 0 0 0 0 0 0 + + + + diff --git a/exercises/ardrone_navigation/ardrone_conf.yml b/exercises/ardrone_navigation/ardrone_conf.yml new file mode 100644 index 000000000..64e235fb8 --- /dev/null +++ b/exercises/ardrone_navigation/ardrone_conf.yml @@ -0,0 +1,38 @@ +Introrob: + + Camera: + Server: 1 # 0 -> Deactivate, 1 -> Ice , 2 -> ROS + Proxy: "Camera:tcp -h localhost -p 9000" + Format: RGB8 + Topic: "/IntrorobROS/image_raw" + Name: IntrorobCamera + + Pose3D: + Server: 1 # 0 -> Deactivate, 1 -> Ice , 2 -> ROS + Proxy: "Pose3D:tcp -h localhost -p 9000" + Topic: "/IntrorobROS/Pose3D" + Name: IntrorobPose3d + + CMDVel: + Server: 1 # 0 -> Deactivate, 1 -> Ice , 2 -> ROS + Proxy: "CMDVel:tcp -h localhost -p 9000" + Topic: "/IntrorobROS/CMDVel" + Name: IntrorobCMDVel + + Navdata: + Server: 1 # 0 -> Deactivate, 1 -> Ice , 2 -> ROS + Proxy: "Navdata:tcp -h localhost -p 9000" + Topic: "/IntrorobROS/Navdata" + Name: IntrorobNavdata + + Extra: + Server: 1 # 0 -> Deactivate, 1 -> Ice , 2 -> ROS + Proxy: "Extra:tcp -h localhost -p 9000" + Topic: "/IntrorobROS/Extra" + Name: IntrorobExtra + + Xmax: 1 + Ymax: 1 + Zmax: 1 + Yawmax: 1 +NodeName: Introrob diff --git a/exercises/ardrone_navigation/generateGUI b/exercises/ardrone_navigation/generateGUI new file mode 100755 index 000000000..d2d26b788 --- /dev/null +++ b/exercises/ardrone_navigation/generateGUI @@ -0,0 +1,9 @@ +#!/bin/bash + cd resources + pyrcc5 resources.qrc -o resources_rc.py + mv resources_rc.py .. + + cd ../gui + pyuic5 ui_gui.ui > ui_gui.py + cd .. + diff --git a/exercises/ardrone_navigation/gui/GUI.py b/exercises/ardrone_navigation/gui/GUI.py new file mode 100644 index 000000000..e4f7466df --- /dev/null +++ b/exercises/ardrone_navigation/gui/GUI.py @@ -0,0 +1,202 @@ +# +# Copyright (C) 1997-2016 JDE Developers Team +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# Authors : +# Alberto Martin Florido +# Aitor Martinez Fernandez +# + + +from PyQt5.QtCore import pyqtSignal, Qt +from PyQt5 import QtCore, QtGui, QtWidgets +from PyQt5.QtWidgets import QMainWindow +from gui.ui_gui import Ui_MainWindow +from gui.teleopWidget import TeleopWidget +from gui.cameraWidget import CameraWidget +from gui.communicator import Communicator +from gui.sensorsWidget import SensorsWidget +from gui.logoWidget import LogoWidget + +class MainWindow(QMainWindow, Ui_MainWindow): + + updGUI=pyqtSignal() + def __init__(self, parent=None): + super(MainWindow, self).__init__(parent) + self.setupUi(self) + self.teleop=TeleopWidget(self) + self.tlLayout.addWidget(self.teleop) + self.teleop.setVisible(True) + + self.logo = LogoWidget(self, self.logoLayout.parent().width(), self.logoLayout.parent().height()) + self.logoLayout.addWidget(self.logo) + self.logo.setVisible(True) + + self.record = False + + self.updGUI.connect(self.updateGUI) + + self.cameraCheck.stateChanged.connect(self.showCameraWidget) + self.sensorsCheck.stateChanged.connect(self.showSensorsWidget) + + self.rotationDial.valueChanged.connect(self.rotationChange) + self.altdSlider.valueChanged.connect(self.altitudeChange) + + self.cameraWidget=CameraWidget(self) + self.sensorsWidget=SensorsWidget(self) + + self.cameraCommunicator=Communicator() + self.trackingCommunicator = Communicator() + + #self.stopButton.clicked.connect(self.stopClicked) + self.playButton.clicked.connect(self.playClicked) + self.playButton.setCheckable(True) + self.resetButton.clicked.connect(self.resetClicked) + self.takeoffButton.clicked.connect(self.takeOffClicked) + self.takeoff=False + self.reset=False + + def getCamera(self): + return self.camera + + def setCamera(self,camera): + self.camera = camera + + def getNavData(self): + return self.navdata + + def setNavData(self,navdata): + self.navdata = navdata + + def getPose3D(self): + return self.pose + + def setPose3D(self,pose): + self.pose = pose + + def getCMDVel(self): + return self.cmdvel + + def setCMDVel(self,cmdvel): + self.cmdvel = cmdvel + + def getExtra(self): + return self.extra + + def setExtra(self,extra): + self.extra = extra + + def setAlgorithm(self, algorithm ): + self.algorithm=algorithm + + def getAlgorithm(self): + return self.algorithm + + def updateGUI(self): + self.cameraWidget.imageUpdate.emit() + self.sensorsWidget.sensorsUpdate.emit() + + def playClicked(self): + if self.playButton.isChecked(): + if self.record == True: + self.extra.record(True) + icon = QtGui.QIcon() + self.playButton.setText("Stop Code") + self.playButton.setStyleSheet("background-color: #ec7063") + icon.addPixmap(QtGui.QPixmap(":/images/stop.png"), QtGui.QIcon.Normal, QtGui.QIcon.Off) + self.playButton.setIcon(icon) + self.algorithm.play() + else: + if self.record == True: + self.extra.record(False) + icon = QtGui.QIcon() + self.playButton.setStyleSheet("background-color: #7dcea0") + icon.addPixmap(QtGui.QPixmap(":/images/play.png"), QtGui.QIcon.Normal, QtGui.QIcon.Off) + self.playButton.setIcon(icon) + self.playButton.setText("Play Code") + self.algorithm.stop() + + #def stopClicked(self): + # if self.record == True: + # self.extra.record(False) + # self.algorithm.stop() + # self.rotationDial.setValue(self.altdSlider.maximum()/2) + # self.altdSlider.setValue(self.altdSlider.maximum()/2) + # self.cmdvel.sendCMDVel(0,0,0,0,0,0) + # self.teleop.stopSIG.emit()''' + + def takeOffClicked(self): + if(self.takeoff==True): + self.takeoffButton.setText("Take Off") + self.extra.land() + self.takeoff=False + else: + self.takeoffButton.setText("Land") + self.extra.takeoff() + self.takeoff=True + + def resetClicked(self): + if self.reset == True: + self.resetButton.setText("Reset") + self.extra.reset() + self.reset=False + else: + self.resetButton.setText("Unreset") + self.extra.reset() + self.reset=True + + def showCameraWidget(self,state): + if state == Qt.Checked: + self.cameraWidget.show() + else: + self.cameraWidget.close() + + def closeCameraWidget(self): + self.cameraCheck.setChecked(False) + + def showSensorsWidget(self,state): + if state == Qt.Checked: + self.sensorsWidget.show() + else: + self.sensorsWidget.close() + + def closeSensorsWidget(self): + self.sensorsCheck.setChecked(False) + + def rotationChange(self,value): + value=(1.0/(self.rotationDial.maximum()/2))*(value - (self.rotationDial.maximum()/2)) + self.rotValue.setText('%.2f' % value) + self.cmdvel.setYaw(value) + self.cmdvel.sendVelocities() + + def altitudeChange(self,value): + value=(1.0/(self.altdSlider.maximum()/2))*(value - (self.altdSlider.maximum()/2)) + self.altdValue.setText('%.2f' % value) + self.cmdvel.setVZ(value) + self.cmdvel.sendVelocities() + + def setXYValues(self,newX,newY): + self.XValue.setText('%.2f' % newX) + self.YValue.setText('%.2f' % newY) + self.cmdvel.setVX(-newY) + self.cmdvel.setVY(-newX) + self.cmdvel.sendVelocities() + + def closeEvent(self, event): + self.algorithm.kill() + self.camera.stop() + self.navdata.stop() + self.pose.stop() + event.accept() + diff --git a/exercises/ardrone_navigation/gui/__init__.py b/exercises/ardrone_navigation/gui/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/exercises/ardrone_navigation/gui/cameraWidget.py b/exercises/ardrone_navigation/gui/cameraWidget.py new file mode 100644 index 000000000..09111dba6 --- /dev/null +++ b/exercises/ardrone_navigation/gui/cameraWidget.py @@ -0,0 +1,84 @@ +# +# Copyright (C) 1997-2015 JDE Developers Team +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# Authors : +# Alberto Martin Florido +# +from PyQt5.QtCore import QSize, pyqtSignal +from PyQt5.QtGui import QImage, QPixmap +from PyQt5.QtWidgets import QPushButton,QWidget, QLabel + + +class CameraWidget(QWidget): + IMAGE_COLS_MAX=640 + IMAGE_ROWS_MAX=360 + LINX=0.3 + LINY=0.3 + LINZ=0.8 + ANGZ=1.0 + ANGY=0.0 + ANGX=0.0 + + imageUpdate=pyqtSignal() + + def __init__(self,winParent): + super(CameraWidget, self).__init__() + self.winParent=winParent + self.imageUpdate.connect(self.updateImage) + self.initUI() + + def initUI(self): + + self.setMinimumSize(680,500) + self.setMaximumSize(680,500) + + self.setWindowTitle("Camera") + changeCamButton=QPushButton("Change Camera") + changeCamButton.resize(170,40) + changeCamButton.move(245,450) + changeCamButton.setParent(self) + changeCamButton.clicked.connect(self.changeCamera) + + self.imgLabel=QLabel(self) + self.imgLabel.resize(640,360) + self.imgLabel.move(10,5) + self.imgLabel.show() + + def updateImage(self): + + img = self.winParent.getCamera().getImage().data + if img is not None: + image = QImage(img.data, img.shape[1], img.shape[0], img.shape[1]*img.shape[2], QImage.Format_RGB888); + + if img.shape[1]==self.IMAGE_COLS_MAX: + x=20 + else: + x=(self.IMAGE_COLS_MAX+20)/2-(img.shape[1]/2) + if img.shape[0]==self.IMAGE_ROWS_MAX: + y=40 + else: + y=(self.IMAGE_ROWS_MAX+40)/2-(img.shape[0]/2) + + size=QSize(img.shape[1],img.shape[0]) + self.imgLabel.move(x,y) + self.imgLabel.resize(size) + self.imgLabel.setPixmap(QPixmap.fromImage(image)) + + def closeEvent(self, event): + self.winParent.closeCameraWidget() + + def changeCamera(self): + self.winParent.getExtra().toggleCam() + diff --git a/exercises/ardrone_navigation/gui/communicator.py b/exercises/ardrone_navigation/gui/communicator.py new file mode 100644 index 000000000..ac7c1a026 --- /dev/null +++ b/exercises/ardrone_navigation/gui/communicator.py @@ -0,0 +1,24 @@ +# +# Copyright (C) 1997-2015 JDE Developers Team +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# Authors : +# Alberto Martin Florido +# +#!/usr/bin/python +# -*- coding: utf-8 -*- + +from PyQt5 import QtCore +class Communicator(QtCore.QObject): + updateBW = QtCore.pyqtSignal() diff --git a/exercises/ardrone_navigation/gui/logoWidget.py b/exercises/ardrone_navigation/gui/logoWidget.py new file mode 100644 index 000000000..59b6498d8 --- /dev/null +++ b/exercises/ardrone_navigation/gui/logoWidget.py @@ -0,0 +1,44 @@ +# +# Copyright (C) 1997-2015 JDE Developers Team +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# Authors : +# Alberto Martin Florido +# +import resources_rc +from PyQt5 import QtGui +from PyQt5.QtCore import pyqtSignal, QPointF, Qt, QPoint +from PyQt5.QtWidgets import QWidget, QGridLayout + +class LogoWidget(QWidget): + + def __init__(self,winParent, width=0, height=0): + super(LogoWidget, self).__init__() + self.winParent=winParent + qimage=QtGui.QImage() + qimage.load(':images/jderobot.png') + if (width != 0 and height != 0): + self.qimage = qimage.scaled(0.8*width, 0.8*height, Qt.KeepAspectRatio) + #self.qimage = qimage.scaled(0.8*width, 0.8*height) + self.resize(width, height) + else: + self.qimage = qimage + + + def paintEvent(self, e): + + painter=QtGui.QPainter(self) + painter.drawImage(self.width()/2-self.qimage.width()/2, self.height()/2-self.qimage.height()/2, self.qimage) + + diff --git a/exercises/ardrone_navigation/gui/sensorsWidget.py b/exercises/ardrone_navigation/gui/sensorsWidget.py new file mode 100644 index 000000000..ed043f3ee --- /dev/null +++ b/exercises/ardrone_navigation/gui/sensorsWidget.py @@ -0,0 +1,254 @@ +# +# Copyright (C) 1997-2015 JDE Developers Team +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# Authors : +# Alberto Martin Florido +# +from PyQt5.QtWidgets import QWidget, QLabel, QGridLayout, QVBoxLayout, QSpacerItem, QSizePolicy +from PyQt5 import QtCore +from PyQt5.QtCore import pyqtSignal, Qt +# from gui.speedoMeter import SpeedoMeter +# from gui.attitudeIndicator import AttitudeIndicator +from qfi import qfi_ADI, qfi_ALT, qfi_SI, qfi_HSI +import math + + +class SensorsWidget(QWidget): + sensorsUpdate = pyqtSignal() + + def __init__(self, winParent): + super(SensorsWidget, self).__init__() + self.winParent = winParent + self.sensorsUpdate.connect(self.updateSensors) + self.initUI() + + def initUI(self): + + self.mainLayout = QGridLayout() + self.horizonLayout = QVBoxLayout() + self.horizonData = QGridLayout() + self.compassLayout = QVBoxLayout() + self.compassData = QGridLayout() + self.altLayout = QVBoxLayout() + self.altData = QGridLayout() + + self.setMinimumSize(660, 450) + self.setMaximumSize(660, 450) + + self.setWindowTitle("Sensors") + + self.pitchLabel = QLabel('Pitch:', self) + self.pitchValueLabel = QLabel('0', self) + self.pitchValueLabel.setAlignment(Qt.AlignRight | Qt.AlignTrailing | Qt.AlignVCenter) + + self.rollLabel = QLabel('Roll:', self) + self.rollValueLabel = QLabel('0', self) + self.rollValueLabel.setAlignment(Qt.AlignRight | Qt.AlignTrailing | Qt.AlignVCenter) + + self.yawLabel = QLabel('Yaw:', self) + self.yawValueLabel = QLabel('0', self) + self.yawValueLabel.setAlignment(Qt.AlignRight | Qt.AlignTrailing | Qt.AlignVCenter) + + self.altLabel = QLabel('Alt:', self) + self.altValueLabel = QLabel('0', self) + self.altValueLabel.setAlignment(Qt.AlignRight | Qt.AlignTrailing | Qt.AlignVCenter) + + self.pitchgLabel = QLabel("\272", self) + self.rollgLabel = QLabel("\272", self) + self.yawgLabel = QLabel("\272", self) + self.altmLabel = QLabel('m', self) + + hSpacer = QSpacerItem(100, 30, QSizePolicy.Ignored, QSizePolicy.Ignored) + + self.horizonData.addItem(hSpacer, 0, 0, 1, 1, Qt.AlignLeft) + self.horizonData.addWidget(self.pitchLabel, 0, 1, Qt.AlignCenter) + self.horizonData.addWidget(self.pitchValueLabel, 0, 2, Qt.AlignCenter) + self.horizonData.addWidget(self.pitchgLabel, 0, 3, Qt.AlignCenter) + self.horizonData.addWidget(self.rollLabel, 0, 4, Qt.AlignCenter) + self.horizonData.addWidget(self.rollValueLabel, 0, 5, Qt.AlignCenter) + self.horizonData.addWidget(self.rollgLabel, 0, 6, Qt.AlignCenter) + self.horizonData.addItem(hSpacer, 0, 7, 1, 1, Qt.AlignRight) + + self.compassData.addItem(hSpacer, 0, 0, 1, 1, Qt.AlignLeft) + self.compassData.addWidget(self.yawLabel, 0, 1, Qt.AlignCenter) + self.compassData.addWidget(self.yawValueLabel, 0, 2, Qt.AlignCenter) + self.compassData.addWidget(self.yawgLabel, 0, 3, Qt.AlignCenter) + self.compassData.addItem(hSpacer, 0, 4, 1, 1, Qt.AlignRight) + + self.altData.addItem(hSpacer, 0, 0, 1, 1, Qt.AlignLeft) + self.altData.addWidget(self.altLabel, 0, 1, Qt.AlignCenter) + self.altData.addWidget(self.altValueLabel, 0, 2, Qt.AlignCenter) + self.altData.addWidget(self.altmLabel, 0, 3, Qt.AlignCenter) + self.altData.addItem(hSpacer, 0, 4, 1, 1, Qt.AlignLeft) + + self.altd = qfi_ALT.qfi_ALT(self) + + self.altd.setFixedSize(QtCore.QSize(200,200)) + self.altLayout.addWidget(self.altd) + self.altLayout.addLayout(self.altData) + # self.altd.move(420,50) + + self.compass = qfi_HSI.qfi_HSI(self) + self.compass.setFixedSize(QtCore.QSize(200, 200)) + self.compassLayout.addWidget(self.compass) + self.compassLayout.addLayout(self.compassData) + + self.horizon = qfi_ADI.qfi_ADI(self) + self.horizon.setFixedSize(QtCore.QSize(200, 200)) + self.horizonLayout.addWidget(self.horizon) + self.horizonLayout.addLayout(self.horizonData) + + # self.battery=Qwt.QwtThermo(self) + # self.battery.setMaxValue(100.0) + # self.battery.setMinValue(0.0) + # self.battery.setPipeWidth(10) + + # self.battery.move(580,10) + # self.battery.resize(56,241) + # self.batteryLabel=QLabel('Battery (%)',self) + # self.batteryLabel.move(580,251) + + self.velLinX = qfi_SI.qfi_SI(self) + self.velLinX.setFixedSize(QtCore.QSize(150, 150)) + # self.velLinX.move(60,270) + self.velXLabel = QLabel('Linear X (m/s)', self) + # self.velXLabel.move(95,420) + + + self.velLinY = qfi_SI.qfi_SI(self) + self.velLinY.setFixedSize(QtCore.QSize(150, 150)) + # self.velLinY.move(240,270) + self.velYLabel = QLabel('Linear Y (m/s)', self) + # self.velYLabel.move(275,420) + + self.velLinZ = qfi_SI.qfi_SI(self) + self.velLinZ.setFixedSize(QtCore.QSize(150, 150)) + # self.velLinZ.setLabel("8 m/s") + # self.velLinZ.move(420,270) + self.velZLabel = QLabel('Linear Z (m/s)', self) + # self.velZLabel.move(455,420) + + self.mainLayout.addLayout(self.horizonLayout, 0, 0, Qt.AlignCenter) + self.mainLayout.addLayout(self.compassLayout, 0, 1, Qt.AlignCenter) + self.mainLayout.addLayout(self.altLayout, 0, 2, Qt.AlignCenter) + self.mainLayout.addWidget(self.velLinX, 1, 0, Qt.AlignCenter) + self.mainLayout.addWidget(self.velLinY, 1, 1, Qt.AlignCenter) + self.mainLayout.addWidget(self.velLinZ, 1, 2, Qt.AlignCenter) + self.mainLayout.addWidget(self.velXLabel, 2, 0, Qt.AlignCenter) + self.mainLayout.addWidget(self.velYLabel, 2, 1, Qt.AlignCenter) + self.mainLayout.addWidget(self.velZLabel, 2, 2, Qt.AlignCenter) + self.setLayout(self.mainLayout); + + def updateSensors(self): + pose = self.winParent.getPose3D().getPose3d() + + if pose != None: + qw = pose.q[0] + qx = pose.q[1] + qy = pose.q[2] + qz = pose.q[3] + self.drawAltd(pose.z) + self.drawYawValues(self.quatToYaw(qw, qx, qy, qz) * 180 / math.pi) + self.drawPitchRollValues(self.quatToPitch(qw, qx, qy, qz) * 180 / math.pi, + self.quatToRoll(qw, qx, qy, qz) * 180 / math.pi) + + navdata = self.winParent.getNavData().getNavData() + + if navdata != None: + # self.battery.setValue(navdata.batteryPercent) + self.drawVelocities(navdata.vx, navdata.vy, navdata.vz) + + def drawYawValues(self, degress): + value = "{0:.2f}".format(degress) + self.yawValueLabel.setText(value) + self.compass.setHeading(degress) + self.compass.viewUpdate.emit() + + def drawAltd(self, meters): + + self.altd.setAltitude(meters * 10) + self.altd.viewUpdate.emit() + + value = "{0:.0f}".format(meters) + self.altValueLabel.setText(value) + + def drawPitchRollValues(self, pitch, roll): + if (pitch > 0 and pitch <= 90): + result = pitch / 90 + result = -result + elif (pitch < 0 and pitch >= -90): + result = pitch / -90 + else: + result = 0.0 + + self.horizon.setPitch(pitch) + self.horizon.setRoll(-roll) + self.horizon.viewUpdate.emit() + pitchValue = "{0:.2f}".format(pitch) + rollValue = "{0:.2f}".format(roll) + self.pitchValueLabel.setText(pitchValue) + self.rollValueLabel.setText(rollValue) + + def drawVelocities(self, vx, vy, vz): + + vx = math.fabs(vx) + vx /= 1000.0 + self.velLinX.setSpeed(vx) + self.velLinX.viewUpdate.emit() + vx = math.fabs(vx) + + vy = math.fabs(vy) + vy /= 1000.0 + self.velLinY.setSpeed(vy) + self.velLinY.viewUpdate.emit() + + vz = math.fabs(vz) + vz /= 1000.0 + self.velLinZ.setSpeed(vz) + self.velLinZ.viewUpdate.emit() + + def quatToRoll(self, qw, qx, qy, qz): + rotateXa0 = 2.0 * (qy * qz + qw * qx) + rotateXa1 = qw * qw - qx * qx - qy * qy + qz * qz + rotateX = 0.0 + + if (rotateXa0 != 0.0 and rotateXa1 != 0.0): + rotateX = math.atan2(rotateXa0, rotateXa1) + + return rotateX + + def quatToPitch(self, qw, qx, qy, qz): + rotateYa0 = -2.0 * (qx * qz - qw * qy) + rotateY = 0.0 + if (rotateYa0 >= 1.0): + rotateY = math.pi / 2.0 + elif (rotateYa0 <= -1.0): + rotateY = -math.pi / 2.0 + else: + rotateY = math.asin(rotateYa0) + + return rotateY + + def quatToYaw(self, qw, qx, qy, qz): + rotateZa0 = 2.0 * (qx * qy + qw * qz) + rotateZa1 = qw * qw + qx * qx - qy * qy - qz * qz + rotateZ = 0.0 + if (rotateZa0 != 0.0 and rotateZa1 != 0.0): + rotateZ = math.atan2(rotateZa0, rotateZa1) + + return rotateZ + + def closeEvent(self, event): + self.winParent.closeSensorsWidget() diff --git a/exercises/ardrone_navigation/gui/teleopWidget.py b/exercises/ardrone_navigation/gui/teleopWidget.py new file mode 100644 index 000000000..02a067fa7 --- /dev/null +++ b/exercises/ardrone_navigation/gui/teleopWidget.py @@ -0,0 +1,113 @@ +# +# Copyright (C) 1997-2015 JDE Developers Team +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# Authors : +# Alberto Martin Florido +# +import resources_rc +from PyQt5.QtGui import QImage, QPainter, QPen +from PyQt5.QtCore import pyqtSignal, QPointF, Qt, QPoint +from PyQt5.QtWidgets import QWidget, QGridLayout + +class TeleopWidget(QWidget): + + stopSIG=pyqtSignal() + + def __init__(self,winParent): + super(TeleopWidget, self).__init__() + self.winParent=winParent + self.line = QPointF(0, 0); + self.qimage=QImage() + self.qimage.load(':images/ball.png') + self.stopSIG.connect(self.stop) + self.initUI() + + def initUI(self): + layout=QGridLayout() + self.setLayout(layout) + self.setAutoFillBackground(True) + p = self.palette() + p.setColor(self.backgroundRole(), Qt.black) + self.setPalette(p) + self.resize(300,300) + self.setMinimumSize(300,300) + + def stop(self): + self.line = QPointF(0, 0); + self.repaint(); + + def mouseMoveEvent(self,e): + if e.buttons() == Qt.LeftButton: + x = e.x()-self.width()/2 + y = e.y()-self.height()/2 + self.line = QPointF(x, y) + self.repaint() + + def paintEvent(self, e): + _width = self.width() + _height = self.height() + + + width = 2 + + painter=QPainter(self) + + pen = QPen(Qt.blue, width) + painter.setPen(pen) + + #Centro del widget + painter.translate(QPoint(_width/2, _height/2)) + + #eje + painter.drawLine(QPointF(-_width, 0), + QPointF( _width, 0)) + + painter.drawLine(QPointF(0, -_height), + QPointF(0, _height)) + + #con el raton + pen = QPen(Qt.red, width) + painter.setPen(pen) + + #Comprobamos que el raton este dentro de los limites + if abs(self.line.x()*2) >= self.size().width(): + if self.line.x()>=0: + self.line.setX(self.size().width()/2) + elif self.line.x()<0: + self.line.setX((-self.size().width()/2)+1) + + if abs(self.line.y()*2) >= self.size().height(): + if self.line.y()>=0: + self.line.setY(self.size().height()/2) + elif self.line.y()<0: + self.line.setY((-self.size().height()/2)+1) + + painter.drawLine(QPointF(self.line.x(), -_height), + QPointF(self.line.x(), _height)) + + painter.drawLine(QPointF(-_width, self.line.y()), + QPointF( _width, self.line.y())) + + #print "x: %f y: %f" % (self.line.x(), self.line.y()) + + v_normalized = (1.0/(self.size().height()/2)) * self.line.y() + v_normalized = float("{0:.2f}".format(v_normalized)) + w_normalized = (1.0/(self.size().width()/2)) * self.line.x() + w_normalized = float("{0:.2f}".format(w_normalized)) + + #print "v: %f w: %f" % (v_normalized,w_normalized) + self.winParent.setXYValues(w_normalized,v_normalized) + painter.drawImage(self.line.x()-self.qimage.width()/2, self.line.y()-self.qimage.height()/2, self.qimage); + diff --git a/exercises/ardrone_navigation/gui/threadGUI.py b/exercises/ardrone_navigation/gui/threadGUI.py new file mode 100644 index 000000000..1c6e80781 --- /dev/null +++ b/exercises/ardrone_navigation/gui/threadGUI.py @@ -0,0 +1,43 @@ +# +# Copyright (C) 1997-2015 JDE Developers Team +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see http://www.gnu.org/licenses/. +# Authors : +# Alberto Martin Florido +# +import threading, time +from datetime import datetime + +time_cycle = 50; + +class ThreadGUI(threading.Thread): + def __init__(self, gui): + self.gui = gui + + threading.Thread.__init__(self) + + def run(self): + + while(True): + + start_time = datetime.now() + self.gui.updGUI.emit() + + finish_Time = datetime.now() + + dt = finish_Time - start_time + ms = (dt.days * 24 * 60 * 60 + dt.seconds) * 1000 + dt.microseconds / 1000.0 + + if(ms < time_cycle): + time.sleep((time_cycle-ms) / 1000.0); \ No newline at end of file diff --git a/exercises/ardrone_navigation/gui/ui_gui.py b/exercises/ardrone_navigation/gui/ui_gui.py new file mode 100644 index 000000000..bff2a6c93 --- /dev/null +++ b/exercises/ardrone_navigation/gui/ui_gui.py @@ -0,0 +1,120 @@ +# -*- coding: utf-8 -*- + +# Form implementation generated from reading ui file 'ui_gui.ui' +# +# Created by: PyQt5 UI code generator 5.5.1 +# +# WARNING! All changes made in this file will be lost! + +from PyQt5 import QtCore, QtGui, QtWidgets + +class Ui_MainWindow(object): + def setupUi(self, MainWindow): + MainWindow.setObjectName("MainWindow") + MainWindow.resize(660, 370) + MainWindow.setMinimumSize(QtCore.QSize(660, 370)) + MainWindow.setMaximumSize(QtCore.QSize(660, 370)) + self.centralwidget = QtWidgets.QWidget(MainWindow) + self.centralwidget.setObjectName("centralwidget") + self.takeoffButton = QtWidgets.QPushButton(self.centralwidget) + self.takeoffButton.setGeometry(QtCore.QRect(470, 30, 161, 41)) + self.takeoffButton.setObjectName("takeoffButton") + self.altdSlider = QtWidgets.QSlider(self.centralwidget) + self.altdSlider.setGeometry(QtCore.QRect(400, 30, 19, 311)) + self.altdSlider.setMaximum(100) + self.altdSlider.setProperty("value", 49) + self.altdSlider.setOrientation(QtCore.Qt.Vertical) + self.altdSlider.setObjectName("altdSlider") + self.playButton = QtWidgets.QPushButton(self.centralwidget) + self.playButton.setGeometry(QtCore.QRect(470, 80, 161, 51)) + icon = QtGui.QIcon() + icon.addPixmap(QtGui.QPixmap(":/images/play.png"), QtGui.QIcon.Normal, QtGui.QIcon.Off) + self.playButton.setIcon(icon) + self.playButton.setObjectName("playButton") + #self.stopButton = QtWidgets.QPushButton(self.centralwidget) + #self.stopButton.setGeometry(QtCore.QRect(560, 80, 71, 51)) + #icon1 = QtGui.QIcon() + #icon1.addPixmap(QtGui.QPixmap(":/images/stop.png"), QtGui.QIcon.Normal, QtGui.QIcon.Off) + #self.stopButton.setIcon(icon1) + #self.stopButton.setObjectName("stopButton") + self.windowsLabel = QtWidgets.QLabel(self.centralwidget) + self.windowsLabel.setGeometry(QtCore.QRect(540, 190, 71, 21)) + self.windowsLabel.setObjectName("windowsLabel") + self.cameraCheck = QtWidgets.QCheckBox(self.centralwidget) + self.cameraCheck.setGeometry(QtCore.QRect(540, 220, 94, 26)) + self.cameraCheck.setObjectName("cameraCheck") + self.sensorsCheck = QtWidgets.QCheckBox(self.centralwidget) + self.sensorsCheck.setGeometry(QtCore.QRect(540, 250, 94, 26)) + self.sensorsCheck.setObjectName("sensorsCheck") + self.altdLabel = QtWidgets.QLabel(self.centralwidget) + self.altdLabel.setGeometry(QtCore.QRect(390, 340, 51, 21)) + self.altdLabel.setObjectName("altdLabel") + self.verticalLayoutWidget = QtWidgets.QWidget(self.centralwidget) + self.verticalLayoutWidget.setGeometry(QtCore.QRect(20, 30, 361, 301)) + self.verticalLayoutWidget.setObjectName("verticalLayoutWidget") + self.tlLayout = QtWidgets.QVBoxLayout(self.verticalLayoutWidget) + self.tlLayout.setObjectName("tlLayout") + self.rotationDial = QtWidgets.QDial(self.centralwidget) + self.rotationDial.setGeometry(QtCore.QRect(440, 220, 50, 64)) + self.rotationDial.setMaximum(100) + self.rotationDial.setProperty("value", 49) + self.rotationDial.setObjectName("rotationDial") + self.rotationLabel = QtWidgets.QLabel(self.centralwidget) + self.rotationLabel.setGeometry(QtCore.QRect(440, 280, 65, 21)) + self.rotationLabel.setObjectName("rotationLabel") + self.XLabel = QtWidgets.QLabel(self.centralwidget) + self.XLabel.setGeometry(QtCore.QRect(20, 340, 21, 21)) + self.XLabel.setObjectName("XLabel") + self.YLabel = QtWidgets.QLabel(self.centralwidget) + self.YLabel.setGeometry(QtCore.QRect(130, 340, 21, 21)) + self.YLabel.setObjectName("YLabel") + self.XValue = QtWidgets.QLabel(self.centralwidget) + self.XValue.setGeometry(QtCore.QRect(40, 340, 41, 21)) + self.XValue.setAlignment(QtCore.Qt.AlignRight|QtCore.Qt.AlignTrailing|QtCore.Qt.AlignVCenter) + self.XValue.setObjectName("XValue") + self.YValue = QtWidgets.QLabel(self.centralwidget) + self.YValue.setGeometry(QtCore.QRect(150, 340, 41, 21)) + self.YValue.setAlignment(QtCore.Qt.AlignRight|QtCore.Qt.AlignTrailing|QtCore.Qt.AlignVCenter) + self.YValue.setObjectName("YValue") + self.altdValue = QtWidgets.QLabel(self.centralwidget) + self.altdValue.setGeometry(QtCore.QRect(390, 10, 41, 21)) + self.altdValue.setAlignment(QtCore.Qt.AlignCenter) + self.altdValue.setObjectName("altdValue") + self.rotValue = QtWidgets.QLabel(self.centralwidget) + self.rotValue.setGeometry(QtCore.QRect(445, 200, 41, 21)) + self.rotValue.setAlignment(QtCore.Qt.AlignCenter) + self.rotValue.setObjectName("rotValue") + self.resetButton = QtWidgets.QPushButton(self.centralwidget) + self.resetButton.setGeometry(QtCore.QRect(470, 140, 161, 41)) + self.resetButton.setObjectName("resetButton") + self.verticalLayoutWidget_2 = QtWidgets.QWidget(self.centralwidget) + self.verticalLayoutWidget_2.setGeometry(QtCore.QRect(570, 280, 81, 80)) + self.verticalLayoutWidget_2.setObjectName("verticalLayoutWidget_2") + self.logoLayout = QtWidgets.QVBoxLayout(self.verticalLayoutWidget_2) + self.logoLayout.setSpacing(0) + self.logoLayout.setObjectName("logoLayout") + MainWindow.setCentralWidget(self.centralwidget) + + self.retranslateUi(MainWindow) + QtCore.QMetaObject.connectSlotsByName(MainWindow) + + def retranslateUi(self, MainWindow): + _translate = QtCore.QCoreApplication.translate + MainWindow.setWindowTitle(_translate("MainWindow", "Position Control")) + self.takeoffButton.setText(_translate("MainWindow", "Take off")) + self.playButton.setText(_translate("MainWindow", "Play Code")) + #self.stopButton.setText(_translate("MainWindow", "Stop Code")) + self.windowsLabel.setText(_translate("MainWindow", "Windows:")) + self.cameraCheck.setText(_translate("MainWindow", "Camera")) + self.sensorsCheck.setText(_translate("MainWindow", "Sensors")) + self.altdLabel.setText(_translate("MainWindow", "Altitude")) + self.rotationLabel.setText(_translate("MainWindow", "Rotation")) + self.XLabel.setText(_translate("MainWindow", "X:")) + self.YLabel.setText(_translate("MainWindow", "Y:")) + self.XValue.setText(_translate("MainWindow", "0")) + self.YValue.setText(_translate("MainWindow", "0")) + self.altdValue.setText(_translate("MainWindow", "0")) + self.rotValue.setText(_translate("MainWindow", "0")) + self.resetButton.setText(_translate("MainWindow", "Reset")) + +import resources_rc diff --git a/exercises/ardrone_navigation/gui/ui_gui.ui b/exercises/ardrone_navigation/gui/ui_gui.ui new file mode 100644 index 000000000..3ec5a3b61 --- /dev/null +++ b/exercises/ardrone_navigation/gui/ui_gui.ui @@ -0,0 +1,311 @@ + + + MainWindow + + + + 0 + 0 + 660 + 370 + + + + + 660 + 370 + + + + + 660 + 370 + + + + Position Control + + + + + + 470 + 30 + 161 + 41 + + + + Take off + + + + + + 400 + 30 + 19 + 311 + + + + 100 + + + 49 + + + Qt::Vertical + + + + + + 470 + 80 + 71 + 51 + + + + Play + + + + :/images/play.png:/images/play.png + + + + + + 560 + 80 + 71 + 51 + + + + Stop + + + + :/images/stop.png:/images/stop.png + + + + + + 540 + 190 + 71 + 21 + + + + Windows: + + + + + + 540 + 220 + 94 + 26 + + + + Camera + + + + + + 540 + 250 + 94 + 26 + + + + Sensors + + + + + + 390 + 340 + 51 + 21 + + + + Altitude + + + + + + 20 + 30 + 361 + 301 + + + + + + + + 440 + 220 + 50 + 64 + + + + 100 + + + 49 + + + + + + 440 + 280 + 65 + 21 + + + + Rotation + + + + + + 20 + 340 + 21 + 21 + + + + X: + + + + + + 130 + 340 + 21 + 21 + + + + Y: + + + + + + 40 + 340 + 41 + 21 + + + + 0 + + + Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter + + + + + + 150 + 340 + 41 + 21 + + + + 0 + + + Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter + + + + + + 390 + 10 + 41 + 21 + + + + 0 + + + Qt::AlignCenter + + + + + + 445 + 200 + 41 + 21 + + + + 0 + + + Qt::AlignCenter + + + + + + 470 + 140 + 161 + 41 + + + + Reset + + + + + + 570 + 280 + 81 + 80 + + + + + 0 + + + + + + + + + + diff --git a/exercises/ardrone_navigation/map_image.png b/exercises/ardrone_navigation/map_image.png new file mode 100644 index 000000000..b1bfed830 Binary files /dev/null and b/exercises/ardrone_navigation/map_image.png differ diff --git a/exercises/ardrone_navigation/models/inHouse/model.config b/exercises/ardrone_navigation/models/inHouse/model.config new file mode 100644 index 000000000..2fc3fd621 --- /dev/null +++ b/exercises/ardrone_navigation/models/inHouse/model.config @@ -0,0 +1,19 @@ + + + + inHouse + 1.3 + model.sdf + + + Hanqing Xie + + + + + inHouse jderobto teaching + + + + + diff --git a/exercises/ardrone_navigation/models/inHouse/model.sdf b/exercises/ardrone_navigation/models/inHouse/model.sdf new file mode 100644 index 000000000..a7ef44189 --- /dev/null +++ b/exercises/ardrone_navigation/models/inHouse/model.sdf @@ -0,0 +1,877 @@ + + + + true + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + 10 10 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + 10 0 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + 10 -10 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + 5 10 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + 5 5 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + 5 0 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + 0 10 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + 0 5 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + 0 0 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + -5 0 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + -5 -5 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + -5 -10 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + -10 10 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + -10 0 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + -10 -5 0 0 0 0 + + + + + + 0.2 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 0.2 4.0 + + + + + + + + 0 + 0 + + -10 -10 0 0 0 0 + + + + + + + 5 10 0.05 + + + 0 0 0.025 0 0 0 + + + 0 0 0.025 0 0 0 + + + 5 10 0.05 + + + + + + + + 0 + 0 + + 7.5 5 4.0 0 0 0 + + + + + + 5 5 0.05 + + + 0 0 0.025 0 0 0 + + + 0 0 0.025 0 0 0 + + + 5 5 0.05 + + + + + + + + 0 + 0 + + 2.5 7.5 4.0 0 0 0 + + + + + + 10 10 0.05 + + + 0 0 0.025 0 0 0 + + + 0 0 0.025 0 0 0 + + + 10 10 0.05 + + + + + + + + 0 + 0 + + -5 5 4.0 0 0 0 + + + + + + 15 10 0.05 + + + 0 0 0.025 0 0 0 + + + 0 0 0.025 0 0 0 + + + 15 10 0.05 + + + + + + + + 0 + 0 + + 2.5 -5 4.0 0 0 0 + + + + + + 5 5 0.05 + + + 0 0 0.025 0 0 0 + + + 0 0 0.025 0 0 0 + + + 5 5 0.05 + + + + + + + + 0 + 0 + + -7.5 -7.5 4.0 0 0 0 + + + + + + 0.2 20 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 20 4.0 + + + + + + + + 0 + 0 + + 10 0 4.0 0 0 0 + + + + + + 20 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 20 0.2 4.0 + + + + + + + + 0 + 0 + + 0 10 4.0 0 0 0 + + + + + + 0.2 20 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 20 4.0 + + + + + + + + 0 + 0 + + -10 0 4.0 0 0 0 + + + + + + 20 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 20 0.2 4.0 + + + + + + + + 0 + 0 + + 0 -10 4.0 0 0 0 + + + + + + 5 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 5 0.2 4.0 + + + + + + + + 0 + 0 + + 7.5 -5 4.0 0 0 0 + + + + + + 0.2 10 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 10 4.0 + + + + + + + + 0 + 0 + + 5 0 4.0 0 0 0 + + + + + + 0.2 15 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 15 4.0 + + + + + + + + 0 + 0 + + 0 2.5 4.0 0 0 0 + + + + + + 5 0.2 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 5 0.2 4.0 + + + + + + + + 0 + 0 + + -7.5 0 4.0 0 0 0 + + + + + + 0.2 5 4.0 + + + 0 0 2.0 0 0 0 + + + 0 0 2.0 0 0 0 + + + 0.2 5 4.0 + + + + + + + + 0 + 0 + + -5 -7.5 4.0 0 0 0 + + + \ No newline at end of file diff --git a/exercises/ardrone_navigation/navigation/__init__.py b/exercises/ardrone_navigation/navigation/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/exercises/ardrone_navigation/navigation/control/__init__.py b/exercises/ardrone_navigation/navigation/control/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/exercises/ardrone_navigation/navigation/control/ardroneControl.py b/exercises/ardrone_navigation/navigation/control/ardroneControl.py new file mode 100644 index 000000000..3c014218c --- /dev/null +++ b/exercises/ardrone_navigation/navigation/control/ardroneControl.py @@ -0,0 +1,88 @@ +import numpy as np +import time +import math + +class ardroneControl: + def __init__(self, show = False): + #self.pathlist = pathlist + self.getTarget = False + self.controlYaw = False + self.getFinal = False + self.show = show + self.PathId = 0 + + def restart(self): + self.getTarget = False + self.controlYaw = False + self.getFinal = False + self.PathId = 0 + + def setPath(self, pathlist): + self.pathlist = pathlist + + def getPath(self): + return self.pathlist + + def setControlYaw(self, controlYaw): + #self.controlYaw = False + self.controlYaw = controlYaw + + def findTarget(self,Pose): + X = Pose[0] + Y = Pose[1] + Z = Pose[2] + num = len(self.pathlist[0])-1 + TargetNowX = self.pathlist[0][self.PathId] + TargetNowY = self.pathlist[1][self.PathId] + TargetNowZ = self.pathlist[2][self.PathId] + + dis = math.sqrt(pow(X-TargetNowX,2)+pow(Y-TargetNowY,2)+pow(Z-TargetNowZ,2)) + if dis < 1: + if num == self.PathId: + if dis < 0.2: + self.getTarget = True + else: + self.PathId += 1 + #elif(math.fabs(self.e[0]) < 0.01 and math.fabs(self.e[1])<1.5 and math.fabs(self.e[2])<0.01): + # self.PathId += 1 + print ("path",self.PathId) + target = [0, 0, 0] + target[0] = self.pathlist[0][self.PathId] + target[1] = self.pathlist[1][self.PathId] + target[2] = self.pathlist[2][self.PathId] + return target + + def isGetTarget(self): + return self.getTarget + + def control(self, Pose): + # id = 1, pid + # id = 2, direct + TargetPose = self.findTarget(Pose) + if self.show: + print ("pose", Pose) + print ("target", TargetPose) + + vx = TargetPose[0] - Pose[0] + vy = TargetPose[1] - Pose[1] + vz = TargetPose[2] - Pose[2] + az = 0 - Pose[3] + + vx = self.limiting(vx, 1, -1) + vy = self.limiting(vy, 1, -1) + vz = self.limiting(vz, 1, -1) + az = self.limiting(az, 1, -1) + + if self.show: + print("v", vx,vy,vz,az) + + return [vx,vy,vz,az] + + def limiting(self, num, max, min): + if num > max: + num = max + if num < min: + num = min + + return num + diff --git a/exercises/ardrone_navigation/navigation/control/noHolomonicControl.py b/exercises/ardrone_navigation/navigation/control/noHolomonicControl.py new file mode 100644 index 000000000..de1c4104e --- /dev/null +++ b/exercises/ardrone_navigation/navigation/control/noHolomonicControl.py @@ -0,0 +1,159 @@ +import numpy as np +import time +import math + +class noHolomonicControl: + def __init__(self, show = False): + #self.pathlist = pathlist + self.find_path = False + self.PathId = 0 + self.getTarget = False + self.v_old = 0 + self.w_old = 0 + self.v = 0 + self.w = 0 + self.e_old = [0,0,0] + self.e = [0,0,0] + self.controlYaw = True + self.getFinal = False + self.show = show + + def restart(self): + #self.pathlist = pathlist + self.find_path = False + self.PathId = 0 + self.getTarget = False + self.v_old = 0 + self.w_old = 0 + self.v = 0 + self.w = 0 + self.e_old = [0,0,0] + self.e = [0,0,0] + self.controlYaw = True + self.getFinal = False + self.show = show + + def setPath(self, pathlist): + self.pathlist = pathlist + + def getPath(self): + return self.pathlist + + def setControlYaw(self, controlYaw): + #self.controlYaw = False + self.controlYaw = controlYaw + + def findTarget(self,Pose,id): + X = Pose[0] + Y = Pose[1] + Yaw = Pose[2] + num = len(self.pathlist[0])-1 + TargetNowX = self.pathlist[0][self.PathId] + TargetNowY = self.pathlist[1][self.PathId] + dis = math.sqrt(pow(X-TargetNowX,2)+pow(Y-TargetNowY,2)) + if dis < 1: + if num == self.PathId: + if id==1: + if math.fabs(self.e[0]) < 0.1 and math.fabs(self.e[1])<0.5 and math.fabs(self.e[2])<0.1: + self.getTarget = True + else: + self.PathId += 1 + #elif(math.fabs(self.e[0]) < 0.01 and math.fabs(self.e[1])<1.5 and math.fabs(self.e[2])<0.01): + # self.PathId += 1 + + print ("path",self.PathId) + target = [X,Y,Yaw] + target[0] = self.pathlist[0][self.PathId] + target[1] = self.pathlist[1][self.PathId] + target[2] = self.pathlist[2][self.PathId] + return target + + def isGetTarget(self): + return self.getTarget + + def control(self, Pose, id): + # id = 1, pid + # id = 2, direct + TargetPose = self.findTarget(Pose,id) + num = len(self.pathlist[0])-1 + if self.show: + print ("pose", Pose) + print ("target", TargetPose) + if id==1: + self.e[0] = math.cos(Pose[2])*(TargetPose[0] - Pose[0]) + math.sin(Pose[2])*(TargetPose[1] - Pose[1]) + self.e[1] = -math.sin(Pose[2])*(TargetPose[0] - Pose[0]) + math.cos(Pose[2])*(TargetPose[1] - Pose[1]) + self.e[2] = TargetPose[2] - Pose[2] + #print ("e:",self.e) + #print ("e_old:",self.e_old) + self.v = self.v_old + 0.2*(self.e[0] - self.e_old[0]) + 0.01*self.e[0] + self.w = self.w_old + 0.5*self.v*(self.e[1] - self.e_old[1]) + 0.001*self.e[1] + 0.2*(self.e[2] - self.e_old[2]) + 0.01*self.e[2] + #self.pathlist + #print ("v w: ",self.v,self.w) + #print ("v w old: ",self.v_old,self.w_old) + if self.v > 0.5: + self.w = self.w*0.5/self.v + self.v = 0.5 + if self.v < -0.5: + self.w = self.w*-0.5/self.v + self.v = -0.5 + self.v_old = self.v + self.w_old = self.w + self.e_old[0] = self.e[0] + self.e_old[1] = self.e[1] + self.e_old[2] = self.e[2] + else: + if self.controlYaw: + theta = math.atan2((TargetPose[1] - Pose[1]),(TargetPose[0] - Pose[0])) + theta_01 = theta - Pose[2] + theta_21 = TargetPose[2] - theta + theta_20 = TargetPose[2] - Pose[2] + dis = math.sqrt(math.pow((TargetPose[1] - Pose[1]),2)+math.pow((TargetPose[0] - Pose[0]),2)) + if self.show: + + print ("theta_01 ",theta_01) + print ("theta_21 ",theta_21) + print ("dis ",dis) + if math.fabs(theta_01) > 0.2 and dis > 0.1 and ~self.getFinal: + self.v = 0 + if theta_01 > 0: + self.w = max(0.2, theta_01) + else: + self.w = min(-0.2, theta_01) + elif dis > 0.1 and ~self.getFinal: + self.v = max(0.2, dis) + self.w = 0 + elif math.fabs(theta_20) > 0.2 and num == self.PathId: + self.v = 0 + if theta_21 > 0: + self.w = max(0.2, theta_20) + else: + self.w = min(-0.2, theta_20) + self.getFinal = True + # elif math.fabs(theta_21) > 0.2: + # self.v = 0 + # if theta_21 > 0: + # self.w = max(0.2, theta_21) + # else: + # self.w = min(-0.2, theta_21) + else: + print ("get target") + self.v = 0 + self.w = 0 + else: + theta = math.atan2((TargetPose[1] - Pose[1]),(TargetPose[0] - Pose[0])) + theta_01 = theta - Pose[2] + dis = math.sqrt(math.pow((TargetPose[1] - Pose[1]),2)+math.pow((TargetPose[0] - Pose[0]),2)) + if math.fabs(theta_01) > 0.05: + self.v = 0 + self.w = 0.5*theta_01 + elif dis > 0.1: + self.v = max(0.1, 0.5*dis) + self.w = 0 + else: + print ("get target") + self.v = 0 + self.w = 0 + if self.show: + print ('v, w ', self.v, self.w) + + return [self.v, self.w] diff --git a/exercises/ardrone_navigation/navigation/laser/__init__.py b/exercises/ardrone_navigation/navigation/laser/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/exercises/ardrone_navigation/navigation/laser/laser_handle.py b/exercises/ardrone_navigation/navigation/laser/laser_handle.py new file mode 100644 index 000000000..6ff1b2d7d --- /dev/null +++ b/exercises/ardrone_navigation/navigation/laser/laser_handle.py @@ -0,0 +1,59 @@ +import numpy as np +import math +from math import cos,sin,tan,pow,pi,sqrt + +class laser_handle: + def autopark_place(self, laser_data, pose): + #assert len(laser_data) == len(laser_angle) + assert len(pose) == 3 + car_x = pose[0] + car_y = pose[1] + car_yaw = pose[2] + num = len(laser_data) + tmp = 0 + right_angle = 90 + left_angle = 90 + for i in range(num/2): + if laser_data[num/2 - i][0] == 10: + tmp += 1 + right_angle = 90 - i + else: + # print ("laser_data",laser_data[num/2 - i]) + break + + + for i in range(num/2): + if laser_data[num/2 + i][0] == 10: + tmp += 1 + left_angle = 90 + i + else: + # print ("laser_data",laser_data[num/2 - i]) + break + if right_angle == left_angle: + print ("not free place") + return None + else: + print ("left_angle",left_angle,"right_angle",right_angle) + len_angle = left_angle - right_angle + print ("len_angle ",len_angle, ", tmp_length = ",tmp) + max_length = sqrt(pow(10,2)-pow(5,2)) + left_park_len = min(tan((left_angle - 90)*pi/180) * 5.5, max_length) + right_park_len = min(tan((90 - right_angle)*pi/180) * 5.5, max_length) + park_len = left_park_len + right_park_len + print ("park_len",park_len) + + if park_len > 8: + print ("find park place") + car_head_len = (left_park_len-right_park_len)/2 + print ("car_head_len",car_head_len) + print ("pose",pose) + x_tmp = 5.5*sin(car_yaw) + car_head_len * cos(car_yaw) + y_tmp = - 5.5*cos(car_yaw) + car_head_len * sin(car_yaw) + + park_x = x_tmp + car_x + park_y = y_tmp + car_y + park_yaw = car_yaw + print ("park place = ", park_x, park_y, park_yaw) + return [park_x, park_y, park_yaw] + else: + return None diff --git a/exercises/ardrone_navigation/navigation/map/__init__.py b/exercises/ardrone_navigation/navigation/map/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/exercises/ardrone_navigation/navigation/map/costmap_2d.py b/exercises/ardrone_navigation/navigation/map/costmap_2d.py new file mode 100644 index 000000000..648057727 --- /dev/null +++ b/exercises/ardrone_navigation/navigation/map/costmap_2d.py @@ -0,0 +1,322 @@ +import numpy as np +import math +from math import cos,sin,tan,pow,pi +import cv2 + +class costmap_2d: + def __init__(self,cells_size_x, cells_size_y, resolution, origin_x = 0, origin_y = 0, default_value = 0): + self.size_x_ = cells_size_x + self.size_y_ = cells_size_y + self.resolution_ = resolution # grid length + self.origin_x_ = origin_x + self.origin_y_ = origin_y + self.default_value_ = default_value + + + self.initMaps(self.size_x_, self.size_y_,self.default_value_) + + def initMaps(self, size_x, size_y,default_value): + if (size_x>0 and size_y>0): + self.costmap_ = np.full((size_x, size_y), default_value, int) + else: + self.costmap_ = None + print ("map's size error") + + def reinitMap(self, size_x, size_y, resolution, origin_x, origin_y): + self.size_x_ = size_x + self.size_y_ = size_y + self.resolution_ = resolution + self.origin_x_ = origin_x + self.origin_y_ = origin_y + + self.initMaps(self.size_x_, self.size_y_,self.default_value_) + + def resetMap(self, x0, y0, xn, yn): + if xn > self.size_x_: + xn = self.size_x_ + if yn > self.size_y_: + yn = self.size_y_ + lenx = xn - x0 + leny = yn - y0 + for i in range(lenx): + for j in range(leny): + self.costmap_[i+x0,j+y0] = self.default_value_ + + def cellDistance(self, world_dist): + cells_dist = max(0.0, math.ceil(world_dist / self.resolution_)) + return cells_dist + + def getMap(self): + return self.costmap_ + + def getCost(self, mx, my): + if mx < self.size_x_ and my < self.size_y_: + return self.costmap_[mx, my] + else: + return None + + def setCost(self, mx, my, cost): + if mx < self.size_x_ and my < self.size_y_: + self.costmap_[mx, my] = cost + else: + print("Over the border") + + def getSizeInCellsX(self): + return self.size_x_ + + def getSizeInCellsY(self): + return self.size_y_ + + def getSizeInMetersX(self): + return (self.size_x_ - 1 + 0.5) * self.resolution_ + + def getSizeInMetersY(self): + return (self.size_y_ - 1 + 0.5) * self.resolution_ + + def getOriginX(self): + return self.origin_x_ + + def getOriginY(self): + return self.origin_y_ + + def getResolution(self): + return self.resolution_ + + def getSize(self, min_x, min_y, max_x, max_y): + cell_size_x = (int)((max_x - min_x) / self.resolution_) + cell_size_y = (int)((max_y - min_y) / self.resolution_) + return [cell_size_x, cell_size_y] + + def mapToWorld(self, mx, my): + wx = self.origin_x_ + (mx + 0.5) * self.resolution_ + wy = self.origin_y_ + (my + 0.5) * self.resolution_ + return [wx, wy] + + + def worldToMap(self, wx, wy): + if (wx < self.origin_x_ or wy < self.origin_y_): + return False + + mx = (int)((wx - self.origin_x_) / self.resolution_) + my = (int)((wy - self.origin_y_) / self.resolution_) + + if (mx < self.size_x_ and my < self.size_y_): + return [mx, my] + else: + return False + + def worldToMapNoBounds(self, wx, wy): + mx = (int)((wx - self.origin_x_) / self.resolution_) + my = (int)((wy - self.origin_y_) / self.resolution_) + return [mx, my] + + def worldToMapEnforceBounds(self, wx, wy): + # Here we avoid doing any math to wx,wy before comparing them to + # the bounds, so their values can go out to the max and min values + # of double floating point. + if (wx < self.origin_x_): + mx = 0 + elif (wx > self.resolution_ * self.size_x_ + self.origin_x_): + mx = self.size_x_ - 1 + else: + mx = (int)((wx - self.origin_x_) / self.resolution_) + + if (wy < self.origin_y_): + my = 0 + elif (wy > self.resolution_ * self.size_y_ + self.origin_y_): + my = self.size_y_ - 1 + else: + my = (int)((wy - self.origin_y_) / self.resolution_) + return [mx, my] + + + def updateOrigin(self, new_origin_x, new_origin_y): + + cell_ox = int((new_origin_x - self.origin_x_) / self.resolution_) + cell_oy = int((new_origin_y - self.origin_y_) / self.resolution_) + + new_grid_ox = self.origin_x_ + cell_ox * self.resolution_ + new_grid_oy = self.origin_y_ + cell_oy * self.resolution_ + + size_x = self.size_x_ + size_y = self.size_y_ + + lower_left_x = min(max(cell_ox, 0), size_x) + lower_left_y = min(max(cell_oy, 0), size_y) + upper_right_x = min(max(cell_ox + size_x, 0), size_x) + upper_right_y = min(max(cell_oy + size_y, 0), size_y) + + cell_size_x = upper_right_x - lower_left_x + cell_size_y = upper_right_y - lower_left_y + + new_costmap = np.full((self.size_x_,self.size_y_), self.default_value_, int) + + for i in range(cell_size_x): + for j in range(cell_size_y): + tmp_x1 = lower_left_x - cell_ox + i + tmp_y1 = lower_left_y - cell_oy + j + tmp_x2 = lower_left_x + i + tmp_y2 = lower_left_y + j + new_costmap[tmp_x1, tmp_y1] = self.costmap_[tmp_x2, tmp_y2] + + self.costmap_ = new_costmap + self.origin_x_ = new_grid_ox + self.origin_y_ = new_grid_oy + + def updateSize(self, new_size_x, new_size_y): + new_costmap = np.full((new_size_x, new_size_y), self.default_value_, int) + + cell_size_x = min(new_size_x, self.size_x_) + cell_size_y = min(new_size_y, self.size_y_) + + for i in range(cell_size_x): + for j in range(cell_size_y): + new_costmap[i,j] = self.costmap_[i,j] + + self.costmap_ = new_costmap + self.size_x_ = new_size_x + self.size_y_ = new_size_y + + def resizeMap(self, minP, maxP): + # max_cell_size = self.getSize(minP[0], minP[1], maxP[0], maxP[1]) + iMinP = self.worldToMapNoBounds(minP[0], minP[1]) + iMaxP = self.worldToMapNoBounds(maxP[0], maxP[1]) + + if iMinP[0] > 0 or iMinP[1] > 0 or iMaxP[0] < self.size_x_ - 1 or iMaxP[1] < self.size_y_ - 1: + print ("reszie map error") + + new_grid_ox = self.origin_x_ + iMinP[0] * self.resolution_ + new_grid_oy = self.origin_y_ + iMinP[1] * self.resolution_ + + new_size_x = iMaxP[0] - iMinP[0] + new_size_y = iMaxP[1] - iMinP[1] + + self.updateSize(new_size_x,new_size_y) + self.updateOrigin(new_grid_ox, new_grid_oy) + + def resizeMapCheckPoint(self,world_pose): + map_pose = self.worldToMap(world_pose[0], world_pose[1]) + if not map_pose: + minP = self.mapToWorld(0,0) + maxP = self.mapToWorld(self.size_x_ - 1, self.size_y_ - 1) + minP = self.min_pose(world_pose, minP) + maxP = self.max_pose(world_pose, maxP) + self.resizeMap(minP, maxP) + + + def updateWeight(self, lo_max, lo_min): + for i in range(self.size_x_): + for j in range(self.size_y_): + if self.costmap_[i,j] > lo_max: + self.costmap_[i,j] = lo_max + if self.costmap_[i,j] < lo_min: + self.costmap_[i,j] = lo_min + + def convexFillCells(self, lineList): + lenLine = len(lineList) + if (lenLine < 2): + return + + for i in range(lenLine-1): + iLine = self.drawBresenham(lineList[i][0], lineList[i][1], lineList[i+1][0], lineList[i+1][1]) + iLenLine = len(iLine) + for j in range(iLenLine): + x = iLine[j][0] + y = iLine[j][1] + self.costmap_[x,y] += 1 + #def raytraceLine(self, x0, y0, x1, y1): + + def drawBresenham(self, x0, y0, x1, y1): + # x is main axes + if x0>x1: + x0,x1=x1,x0 + y0,y1=y1,y0 + + delta_x = x1 - x0 + delta_y = y1 - y0 + d = 0 + if delta_x == 0: + k = 999999999 + else: + k = float(delta_y) / delta_x + + + x = x0 + y = y0 + line = [] + if k > 1: + while True: + if y > y1: + break + line.append([x,y]) + #self.costmap_[x,y] += 1 + y = y + 1 + d = d + 1 / k + if d > 0.5: + x = x + 1 + d = d - 1 + elif k > 0: + while True: + if x > x1: + break + #self.costmap_[x,y] += 1 + line.append([x,y]) + x = x + 1 + d = d + k + if d > 0.5: + y = y + 1 + d = d - 1 + elif k > -1: + while True: + if x > x1: + break + line.append([x,y]) + #self.costmap_[x,y] += 1 + x = x + 1 + d = d - k + if d > 0.5: + y = y - 1 + d = d - 1 + else: + while True: + if y < y1: + break + line.append([x,y]) + #self.costmap_[x,y] += 1 + y = y - 1 + d = d - 1 / k + if d > 0.5: + x = x + 1 + d = d - 1 + return line + + def min_pose(self, p1, p2): + if p1[0] < p2[0]: + p2[0] = p1[0] + if p1[1] < p2[1]: + p2[1] = p1[1] + return p2 + + def max_pose(self, p1, p2): + if p1[0] > p2[0]: + p2[0] = p1[0] + if p1[1] > p2[1]: + p2[1] = p1[1] + return p2 + + def drawMap(self): + h = self.size_x_ + w = self.size_y_ + mapppm = np.zeros((h,w,3)) + + for i in range(h): + for j in range(w): + mapppm[i,j,0] = (self.costmap_[i,j]+ 15)*255/30 + mapppm[i,j,1] = (self.costmap_[i,j]+ 15)*255/30 + mapppm[i,j,2] = (self.costmap_[i,j]+ 15)*255/30 + # if self.map.costmap_[i,j] != 0: + # print self.map.costmap_[i,j] + #print self.map.costmap_ + cv2.imwrite('map_image.ppm',mapppm) + + diff --git a/exercises/ardrone_navigation/navigation/map/costmap_3d.py b/exercises/ardrone_navigation/navigation/map/costmap_3d.py new file mode 100644 index 000000000..82ed5bdca --- /dev/null +++ b/exercises/ardrone_navigation/navigation/map/costmap_3d.py @@ -0,0 +1,298 @@ +import numpy as np +import math +from math import cos,sin,tan,pow,pi +import cv2 +import matplotlib.pyplot as plt +from mpl_toolkits.mplot3d import Axes3D + +class costmap_3d: + def __init__(self,cells_size_x, cells_size_y, cells_sise_z, resolution, origin_x = 0, origin_y = 0,origin_z = 0, default_value = 0): + self.size_x_ = cells_size_x + self.size_y_ = cells_size_y + self.size_z_ = cells_sise_z + self.resolution_ = resolution # grid length + self.origin_x_ = origin_x + self.origin_y_ = origin_y + self.origin_z_ = origin_z + self.default_value_ = default_value + + self.initMaps(self.size_x_, self.size_y_, self.size_z_, self.default_value_) + + def initMaps(self, size_x, size_y, size_z, default_value): + if (size_x>0 and size_y>0 and size_z>0): + self.costmap_ = np.full((size_x, size_y, size_z), default_value, int) + else: + self.costmap_ = None + print ("map's size error") + + def reinitMap(self, size_x, size_y, size_z, resolution, origin_x, origin_y, origin_z): + self.size_x_ = size_x + self.size_y_ = size_y + self.size_z_ = size_z + self.resolution_ = resolution + self.origin_x_ = origin_x + self.origin_y_ = origin_y + self.origin_z_ = origin_z + + self.initMaps(self.size_x_, self.size_y_, self.size_z_, self.default_value_) + + def resetMap(self, x0, y0, z0, xn, yn, zn): + if xn > self.size_x_: + xn = self.size_x_ + if yn > self.size_y_: + yn = self.size_y_ + if zn > self.size_z_: + zn = self.size_z_ + lenx = xn - x0 + leny = yn - y0 + lenz = yn - z0 + for i in range(lenx): + for j in range(leny): + for k in range(lenz): + self.costmap_[i+x0,j+y0,k+z0] = self.default_value_ + + def cellDistance(self, world_dist): + cells_dist = max(0.0, math.ceil(world_dist / self.resolution_)) + return cells_dist + + def getMap(self): + return self.costmap_ + + def getCost(self, mx, my, mz): + if mx < self.size_x_ and my < self.size_y_ and mz < self.size_z_: + return self.costmap_[mx, my, mz] + else: + return None + + def setCost(self, mx, my, mz, cost): + if mx < self.size_x_ and my < self.size_y_ and mz < self.size_z_: + self.costmap_[mx, my, mz] = cost + else: + print("Over the border") + + def getSizeInCellsX(self): + return self.size_x_ + + def getSizeInCellsY(self): + return self.size_y_ + + def getSizeInCellsZ(self): + return self.size_z_ + + def getSizeInMetersX(self): + return (self.size_x_ - 1 + 0.5) * self.resolution_ + + def getSizeInMetersY(self): + return (self.size_y_ - 1 + 0.5) * self.resolution_ + + def getSizeInMetersZ(self): + return (self.size_z_ - 1 + 0.5) * self.resolution_ + + def getOriginX(self): + return self.origin_x_ + + def getOriginY(self): + return self.origin_y_ + + def getOriginZ(self): + return self.origin_z_ + + def getResolution(self): + return self.resolution_ + + def getSize(self, min_x, min_y, min_z, max_x, max_y, max_z): + cell_size_x = (int)((max_x - min_x) / self.resolution_) + cell_size_y = (int)((max_y - min_y) / self.resolution_) + cell_size_z = (int)((max_z - min_z) / self.resolution_) + return [cell_size_x, cell_size_y, cell_size_z] + + def mapToWorld(self, mx, my, mz): + wx = self.origin_x_ + (mx + 0.5) * self.resolution_ + wy = self.origin_y_ + (my + 0.5) * self.resolution_ + wz = self.origin_z_ + (mz + 0.5) * self.resolution_ + return [wx, wy, wz] + + + def worldToMap(self, wx, wy, wz): + if (wx < self.origin_x_ or wy < self.origin_y_ or wz < self.origin_z_): + return False + + mx = (int)((wx - self.origin_x_) / self.resolution_) + my = (int)((wy - self.origin_y_) / self.resolution_) + mz = (int)((wz - self.origin_z_) / self.resolution_) + + if (mx < self.size_x_ and my < self.size_y_ and mz < self.size_z_): + return [mx, my, mz] + else: + return False + + def worldToMapNoBounds(self, wx, wy, wz): + mx = (int)((wx - self.origin_x_) / self.resolution_) + my = (int)((wy - self.origin_y_) / self.resolution_) + mz = (int)((wz - self.origin_z_) / self.resolution_) + return [mx, my, mz] + + def worldToMapEnforceBounds(self, wx, wy, wz): + # Here we avoid doing any math to wx,wy before comparing them to + # the bounds, so their values can go out to the max and min values + # of double floating point. + if (wx < self.origin_x_): + mx = 0 + elif (wx > self.resolution_ * self.size_x_ + self.origin_x_): + mx = self.size_x_ - 1 + else: + mx = (int)((wx - self.origin_x_) / self.resolution_) + + if (wy < self.origin_y_): + my = 0 + elif (wy > self.resolution_ * self.size_y_ + self.origin_y_): + my = self.size_y_ - 1 + else: + my = (int)((wy - self.origin_y_) / self.resolution_) + + if (wz < self.origin_z_): + mz = 0 + elif (wz > self.resolution_ * self.size_z_ + self.origin_z_): + mz = self.size_z_ - 1 + else: + mz = (int)((wz - self.origin_z_) / self.resolution_) + return [mx, my, mz] + + + def updateOrigin(self, new_origin_x, new_origin_y, new_origin_z): + + cell_ox = int((new_origin_x - self.origin_x_) / self.resolution_) + cell_oy = int((new_origin_y - self.origin_y_) / self.resolution_) + cell_oz = int((new_origin_z - self.origin_z_) / self.resolution_) + + new_grid_ox = self.origin_x_ + cell_ox * self.resolution_ + new_grid_oy = self.origin_y_ + cell_oy * self.resolution_ + new_grid_oz = self.origin_z_ + cell_oz * self.resolution_ + + size_x = self.size_x_ + size_y = self.size_y_ + size_z = self.size_z_ + + lower_left_x = min(max(cell_ox, 0), size_x) + lower_left_y = min(max(cell_oy, 0), size_y) + lower_left_z = min(max(cell_oz, 0), size_z) + upper_right_x = min(max(cell_ox + size_x, 0), size_x) + upper_right_y = min(max(cell_oy + size_y, 0), size_y) + upper_right_z = min(max(cell_oz + size_z, 0), size_z) + + cell_size_x = upper_right_x - lower_left_x + cell_size_y = upper_right_y - lower_left_y + cell_size_z = upper_right_z - lower_left_z + + new_costmap = np.full((self.size_x_,self.size_y_,self.size_z_), self.default_value_, int) + + for i in range(cell_size_x): + for j in range(cell_size_y): + for k in range(cell_size_z): + tmp_x1 = lower_left_x - cell_ox + i + tmp_y1 = lower_left_y - cell_oy + j + tmp_z1 = lower_left_z - cell_oz + k + tmp_x2 = lower_left_x + i + tmp_y2 = lower_left_y + j + tmp_z2 = lower_left_z + k + new_costmap[tmp_x1, tmp_y1, tmp_z1] = self.costmap_[tmp_x2, tmp_y2, tmp_z2] + + self.costmap_ = new_costmap + self.origin_x_ = new_grid_ox + self.origin_y_ = new_grid_oy + self.origin_z_ = new_grid_oz + + def updateSize(self, new_size_x, new_size_y, new_size_z): + new_costmap = np.full((new_size_x, new_size_y, new_size_z), self.default_value_, int) + + cell_size_x = min(new_size_x, self.size_x_) + cell_size_y = min(new_size_y, self.size_y_) + cell_size_z = min(new_size_z, self.size_z_) + + for i in range(cell_size_x): + for j in range(cell_size_y): + for k in range(cell_size_z): + new_costmap[i,j,k] = self.costmap_[i,j,k] + + self.costmap_ = new_costmap + self.size_x_ = new_size_x + self.size_y_ = new_size_y + self.size_z_ = new_size_z + + def resizeMap(self, minP, maxP): + # max_cell_size = self.getSize(minP[0], minP[1], maxP[0], maxP[1]) + iMinP = self.worldToMapNoBounds(minP[0], minP[1], minP[2]) + iMaxP = self.worldToMapNoBounds(maxP[0], maxP[1], maxP[2]) + + if iMinP[0] > 0 or iMinP[1] > 0 or iMinP[2] > 0 or iMaxP[0] < self.size_x_ - 1 or iMaxP[1] < self.size_y_ - 1 or iMaxP[2] < self.size_z_ - 1: + print ("reszie map error") + + new_grid_ox = self.origin_x_ + iMinP[0] * self.resolution_ + new_grid_oy = self.origin_y_ + iMinP[1] * self.resolution_ + new_grid_oz = self.origin_z_ + iMinP[2] * self.resolution_ + + new_size_x = iMaxP[0] - iMinP[0] + new_size_y = iMaxP[1] - iMinP[1] + new_size_z = iMaxP[2] - iMinP[2] + + self.updateSize(new_size_x,new_size_y,new_size_z) + self.updateOrigin(new_grid_ox, new_grid_oy,new_grid_oz) + + def resizeMapCheckPoint(self,world_pose): + map_pose = self.worldToMap(world_pose[0], world_pose[1], world_pose[2]) + if not map_pose: + minP = self.mapToWorld(0,0,0) + maxP = self.mapToWorld(self.size_x_ - 1, self.size_y_ - 1, self.size_z_ - 1) + minP = self.min_pose(world_pose, minP) + maxP = self.max_pose(world_pose, maxP) + self.resizeMap(minP, maxP) + + def updateWeight(self, lo_max, lo_min): + for i in range(self.size_x_): + for j in range(self.size_y_): + for k in range(self.size_z_): + if self.costmap_[i,j,k] > lo_max: + self.costmap_[i,j,k] = lo_max + if self.costmap_[i,j,k] < lo_min: + self.costmap_[i,j,k] = lo_min + + def min_pose(self, p1, p2): + if p1[0] < p2[0]: + p2[0] = p1[0] + if p1[1] < p2[1]: + p2[1] = p1[1] + if p1[2] < p2[2]: + p2[2] = p1[2] + return p2 + + def max_pose(self, p1, p2): + if p1[0] > p2[0]: + p2[0] = p1[0] + if p1[1] > p2[1]: + p2[1] = p1[1] + if p1[2] > p2[2]: + p2[2] = p1[2] + return p2 + + def drawMap(self): + lx = self.size_x_ + ly = self.size_y_ + lz = self.size_z_ + + x = np.array([]) + y = np.array([]) + z = np.array([]) + + for i in range(self.size_x_): + for j in range(self.size_y_): + for k in range(self.size_z_): + if self.costmap_[i,j,k] < -5: + x = np.append(x,i) + y = np.append(y,j) + z = np.append(z,k) + + ax = plt.subplot(111, projection='3d') + ax.scatter(x, y, z, c='b') + plt.savefig('map_image.png') + #plt.show() + diff --git a/exercises/ardrone_navigation/navigation/map/occGridMap.py b/exercises/ardrone_navigation/navigation/map/occGridMap.py new file mode 100644 index 000000000..41ae88267 --- /dev/null +++ b/exercises/ardrone_navigation/navigation/map/occGridMap.py @@ -0,0 +1,135 @@ +import numpy as np +import math +from math import cos,sin,tan,pow,pi +from costmap_2d import costmap_2d +from costmap_3d import costmap_3d +import matplotlib.pyplot as plt +from mpl_toolkits.mplot3d import Axes3D +import cv2 + +class occGridMap: + def __init__(self, lo_occ = -10, lo_free = 5,lo_max = 15,lo_min = -15, dimension = 2): + self.dimension = dimension + self.lo_occ = lo_occ + self.lo_free = lo_free + self.lo_max = lo_max + self.lo_min = lo_min + if self.dimension == 2: + self.map = costmap_2d(100, 100, 0.2) + self.m_laserpose = [] + self.m_laserpose.append( [2,0,0] ) + self.m_laserpose.append( [-1.9,0,pi] ) + self.m_laserpose.append( [0.3,-1,-pi/2] ) + self.m_laserMaxRange = 11 + self.m_usableRange = 9 + elif self.dimension == 3: + self.map = costmap_3d(100, 100, 100, 0.2, origin_x = -10, origin_y = -10, origin_z = 0) + #cells_size_x, cells_size_y, cells_sise_z, resolution, origin_x = 0, origin_y = 0,origin_z = 0, default_value = 0 + + def registerScan(self, pose, laserNum, laserData): + laserPose = [0,0,0] + laserPose[0] = pose[0] + cos(pose[2]) * self.m_laserpose[laserNum][0] - sin(pose[2]) * self.m_laserpose[laserNum][1] + laserPose[1] = pose[1] + sin(pose[2]) * self.m_laserpose[laserNum][0] + cos(pose[2]) * self.m_laserpose[laserNum][1] + laserPose[2] = pose[2] + self.m_laserpose[laserNum][2] + + self.computerActiveArea(laserPose, laserData) + + P0 = self.map.worldToMapEnforceBounds(laserPose[0],laserPose[1]) + #print ("P0 = ", P0, laserPose) + + for i in range(180): + r = laserData[i][0] + if r > self.m_laserMaxRange: + continue + if r > self.m_usableRange: + r = self.m_usableRange + phit = [0,0] + angle = math.radians(180 - i) - pi/2 + phit[0] = laserPose[0] + r * cos(laserPose[2] - angle) + phit[1] = laserPose[1] + r * sin(laserPose[2] - angle) + P1 = self.map.worldToMapEnforceBounds(phit[0],phit[1]) + #print ("P1 = ", P1, phit, i, "P0 = ", P0, laserPose) + line = self.map.drawBresenham(P0[0],P0[1],P1[0],P1[1]) + #print ("line = " , line ) + lenLine = len(line) + for i in range(lenLine): + x = line[i][0] + y = line[i][1] + self.map.costmap_[x,y] += self.lo_free + + if r < self.m_usableRange: + self.map.costmap_[P1[0],P1[1]] += self.lo_occ + + self.map.updateWeight(self.lo_max, self.lo_min) + + + def computerActiveArea(self, laserPose, laserData): + #P0 = self.worldToMapNoBounds(laserPose[0], laserPose[1]) + + minP = self.map.mapToWorld(0,0) + maxP = self.map.mapToWorld(self.map.size_x_ - 1, self.map.size_y_ - 1) + + minP = self.map.min_pose(laserPose, minP) + maxP = self.map.max_pose(laserPose, maxP) + + # laser angle 0~180 + for i in range(180): + r = laserData[i][0] + if r > self.m_laserMaxRange: + continue + if r > self.m_usableRange: + r = self.m_usableRange + phit = [0,0] + angle = math.radians(i) - pi/2 + phit[0] = laserPose[0] + r * cos(laserPose[2] - angle) + phit[1] = laserPose[1] + r * sin(laserPose[2] - angle) + #print ("P1 = ", phit, i) + #P1 = self.worldToMapNoBounds(phit[0],phit[1]) + minP = self.map.min_pose(phit, minP) + maxP = self.map.max_pose(phit, maxP) + + + iMinP = self.map.worldToMapNoBounds(minP[0], minP[1]) + iMaxP = self.map.worldToMapNoBounds(maxP[0], maxP[1]) + if iMinP != [0, 0] or iMaxP != [self.map.size_x_ - 1, self.map.size_y_ - 1]: + minP = [minP[0] - 1,minP[1] - 1] + maxP = [maxP[0] + 1,maxP[1] + 1] + self.map.resizeMap(minP, maxP) + #print ("minp = ",minP, "maxP = ", maxP ) + + + def drawMap(self): + if self.dimension == 2: + h = self.map.size_x_ + w = self.map.size_y_ + mapppm = np.zeros((h,w,3)) + + for i in range(h): + for j in range(w): + mapppm[i,j,0] = (self.map.costmap_[i,j]+ 15)*255/30 + mapppm[i,j,1] = (self.map.costmap_[i,j]+ 15)*255/30 + mapppm[i,j,2] = (self.map.costmap_[i,j]+ 15)*255/30 + # if self.map.costmap_[i,j] != 0: + # print self.map.costmap_[i,j] + #print self.map.costmap_ + cv2.imwrite('map_image.ppm',mapppm) + elif self.dimension == 3: + lx = self.map.size_x_ + ly = self.map.size_y_ + lz = self.map.size_z_ + + x = np.array([]) + y = np.array([]) + z = np.array([]) + + for i in range(lx): + for j in range(ly): + for k in range(lz): + if self.map.costmap_[i,j,k] > 0: + x = np.append(x,i) + y = np.append(y,j) + z = np.append(z,k) + + ax = plt.subplot(111, projection='3d') + ax.scatter(x, y, z, c='b') + plt.savefig('map_image.png') diff --git a/exercises/ardrone_navigation/navigation/navigation.py b/exercises/ardrone_navigation/navigation/navigation.py new file mode 100644 index 000000000..eae323270 --- /dev/null +++ b/exercises/ardrone_navigation/navigation/navigation.py @@ -0,0 +1,341 @@ + +from ompl_planner.ompl_planner import ompl_planner +from control.noHolomonicControl import noHolomonicControl +from control.ardroneControl import ardroneControl +from path_processing.smoothPath import smooth +from map.occGridMap import occGridMap,costmap_2d,costmap_3d + +import cv2 +import numpy as np +import matplotlib.pyplot as plt +import math +from math import cos,sin,tan,pow,pi,sqrt +import matplotlib.pyplot as plt +from mpl_toolkits.mplot3d import Axes3D + +class navigation: + def __init__(self, files, origin, resolution, dimension = 2): + + self.smooth = smooth() + #self.occGridMap = occGridMap() + + self.dimension = dimension + origin_x = origin[0] + origin_y = origin[1] + if dimension == 3: + origin_z = origin[2] + + if dimension == 2: + self.control = noHolomonicControl() + if files: + self.have_map = True + self.init_img = cv2.imread(files) + size = self.init_img.shape + self.gridMap = costmap_2d(size[0],size[1],resolution, origin_x, origin_y) + for i in range(size[0]): + for j in range(size[1]): + data = (np.sum(self.init_img[i,j]))/3 + self.gridMap.setCost(i,j,data) + else: + self.have_map = False + self.gridMap = costmap_2d(100,100,resolution, origin_x, origin_y) + + + elif dimension == 3: + self.control = ardroneControl() + self.have_map = False + self.gridMap = costmap_3d(100, 100, 50, 0.2, origin_x = -10, origin_y = -10, origin_z = 0) + #cells_size_x, cells_size_y, cells_sise_z, resolution, origin_x = 0, origin_y = 0,origin_z = 0, default_value = 0 + + self.getTarget = False + + def update_map_autopark(self,car_pose,target_pose): + #car [6,3] + #target [8,5] + p1 = self.move_pose(car_pose,[12,-3]) + p2 = self.move_pose(car_pose,[-12,-3]) + p3 = self.move_pose(car_pose,[12,3]) + p4 = self.move_pose(car_pose,[-12,3]) + p5 = self.move_pose(target_pose,[4,2.5]) + p6 = self.move_pose(target_pose,[-4,2.5]) + + self.gridMap.resizeMapCheckPoint(p1) + self.gridMap.resizeMapCheckPoint(p2) + self.gridMap.resizeMapCheckPoint(p3) + self.gridMap.resizeMapCheckPoint(p4) + self.gridMap.resizeMapCheckPoint(p5) + self.gridMap.resizeMapCheckPoint(p6) + + m_p1 = self.gridMap.worldToMapEnforceBounds(p1[0],p1[1]) + m_p2 = self.gridMap.worldToMapEnforceBounds(p2[0],p2[1]) + m_p3 = self.gridMap.worldToMapEnforceBounds(p3[0],p3[1]) + m_p4 = self.gridMap.worldToMapEnforceBounds(p4[0],p4[1]) + m_p5 = self.gridMap.worldToMapEnforceBounds(p5[0],p5[1]) + m_p6 = self.gridMap.worldToMapEnforceBounds(p6[0],p6[1]) + + print ("getSizeInCellsX",self.gridMap.getSizeInCellsX()) + print ("getSizeInCellsY",self.gridMap.getSizeInCellsY()) + print ("getOriginX",self.gridMap.getOriginX()) + print ("getOriginY",self.gridMap.getOriginY()) + print ("p1",p1,m_p1) + print ("p2",p2,m_p2) + print ("p3",p3,m_p3) + print ("p4",p4,m_p4) + print ("p5",p5,m_p5) + print ("p6",p6,m_p6) + h = self.gridMap.size_x_ + w = self.gridMap.size_y_ + for i in range(h): + for j in range(w): + self.gridMap.setCost(i,j,-10) + + if m_p1[1] == m_p2[1]: + #scene like autopark.world + print ("update map") + num_x_1 = m_p1[0] - m_p2[0] + num_y_1 = m_p1[1] - m_p3[1] + for i in range(num_x_1): + for j in range(num_y_1): + self.gridMap.setCost(i+m_p4[0],j+m_p4[1],15) + num_x_2 = m_p5[0] - m_p6[0] + num_y_2 = m_p3[1] - m_p5[1] + for i in range(num_x_2): + for j in range(num_y_2): + self.gridMap.setCost(i+m_p6[0],j+m_p6[1],15) + self.gridMap.drawMap() + + def move_pose(self,pose,move): + assert len(pose) == 3 + assert len(move) == 2 + dx = move[1]*sin(pose[2]) + move[0]*cos(pose[2]) + dy = -move[1]*cos(pose[2]) + move[0]*sin(pose[2]) + x = pose[0] + dx + y = pose[1] + dy + return [x,y] + + def path_planning_2d(self,current_pose,goal_pose): + startX = current_pose[0] + startY = current_pose[1] + startYaw = current_pose[2] + goalX = goal_pose[0] + goalY = goal_pose[1] + goalYaw = goal_pose[2] + print ("pose",startX, startY, startYaw, goalX, goalY, goalYaw) + + start_pose_map = self.gridMap.worldToMap(startX,startY) + goal_pose_map = self.gridMap.worldToMap(goalX, goalY) + + ompl_control = True + ompl_sol = ompl_planner(self.gridMap, startX, startY, startYaw, goalX, goalY, goalYaw, "rrt", ompl_control, False) + path_list = ompl_sol.omplRunOnce(50) + #print self.pathlist + if path_list : + if ompl_control: + path_list = self.path_smooth(path_list,current_pose) + + self.control.setPath(path_list) + return True + else: + self.control.setControlYaw(False) + self.control.setPath(path_list) + return True + else: + return False + + def path_smooth_2d(self,path_list,current_pose): + plt.figure(1) + plt.plot(path_list[0], path_list[1]) + num = len(path_list[0]) + for i in range(num): + x1 = path_list[0][i] + y1 = path_list[1][i] + yaw = path_list[2][i] + x2 = x1 + math.cos(yaw) + y2 = y1 + math.sin(yaw) + plt.plot([x1,x2],[y1,y2]) + plt.savefig("figure1.jpg") + self.smooth.setPath(path_list) + path_list = self.smooth.Floyd(path_list) + path_list = self.smooth.checkYaw(path_list, current_pose) + + plt.figure(2) + plt.plot(path_list[0], path_list[1]) + + num = len(path_list[0]) + for i in range(num): + x1 = path_list[0][i] + y1 = path_list[1][i] + yaw = path_list[2][i] + x2 = x1 + math.cos(yaw) + y2 = y1 + math.sin(yaw) + plt.plot([x1,x2],[y1,y2]) + plt.savefig("figure2.jpg") + + return path_list + + def path_following_2d(self, current_pose): + if self.getTarget: + print "get Target" + print self.control.getPath() + return [0,0] + + Pose = [0,0,0] + Pose[0] = current_pose[0] + Pose[1] = current_pose[1] + Pose[2] = current_pose[2] + + data = self.control.control(Pose,2) + self.getTarget = self.control.isGetTarget() + return data + + def path_check(self,path_smooth, costmap): + print ('todo') + + def update_map_laser_2d(self, laser, id, pose): + self.occGridMap.registerScan(pose,id,laser) + self.occGridMap.drawMap() + + def update_map_3d(self): + lx = self.gridMap.size_x_ + ly = self.gridMap.size_y_ + lz = self.gridMap.size_z_ + for i in range(lx): + for j in range(ly): + for k in range(lz): + self.gridMap.setCost(i,j,k,15) + + pillar_box = [0.2, 0.2, 4.0] + pillar_swell = [1, 1, 0] + + floor_box = [5,5,0.2] + floor_swell = [1,1,1] + + wall1_box = [5,0.2,5] + wall2_box = [0.2,5,5] + wall_swell = [1,1,0] + + # pillar_pose = [] + # floor_pose = [] + # wall1_pose = [] + # wall2_pose = [] + + for i in range(5): + for j in range(5): + x = i*5-10 + y = j*5-10 + z = 2 + #pillar_pose.append([x,y,z]) + self.add_box_map(pillar_box,[x,y,z],pillar_swell) + + for i in range(4): + for j in range(4): + x = i*5-7.5 + y = j*5-7.5 + z = 4 + if (x==2.5 and y==2.5) or (x==-7.5 and y==-2.5): + print ('no floor') + else: + #floor_pose.append([x,y,z]) + self.add_box_map(floor_box,[x,y,z],floor_swell) + + for i in range(4): + for j in range(2): + x = i*5-7.5 + y = j*20-10 + z = 6.5 + self.add_box_map(wall1_box,[x,y,z],wall_swell) + + for i in range(4): + for j in range(2): + y = i*5-7.5 + x = j*20-10 + z = 6.5 + self.add_box_map(wall2_box,[x,y,z],wall_swell) + + wall1_pose = [[7.5,5,6.5],[-7.5,0,6.5]] + wall2_pose = [[5,2.5,6.5],[5,-2.5,6.5],[0,7.5,6.5],[0,2.5,6.5],[0,-2.5,6.5],[-5,-7.5,6.5]] + + l1 = len(wall1_pose) + l2 = len(wall2_pose) + for i in range(l1): + self.add_box_map(wall1_box,wall1_pose[i],wall_swell) + for i in range(l2): + self.add_box_map(wall2_box,wall2_pose[i],wall_swell) + + self.gridMap.drawMap() + + def add_box_map(self, box, pose, swell): + box_sum = [] + for i,j in zip(box, swell): + summ = i+j + box_sum.append(summ) + + start = [] + for i,j in zip(box_sum, pose): + tmp = j - i/2 + start.append(tmp) + + cell_size = self.gridMap.getSize(0,0,0,box_sum[0],box_sum[1],box_sum[2]) + cell_start = self.gridMap.worldToMapEnforceBounds(start[0],start[1],start[2]) + + for i in range(cell_size[0]): + for j in range(cell_size[1]): + for k in range(cell_size[2]): + self.gridMap.setCost(i+cell_start[0], j+cell_start[1], k+cell_start[2], -10) + + + def path_planning_3d(self,current_pose,goal_pose): + startX = current_pose[0] + startY = current_pose[1] + startZ = current_pose[2] + goalX = goal_pose[0] + goalY = goal_pose[1] + goalZ = goal_pose[2] + print ("pose",startX, startY, startZ, goalX, goalY, goalZ) + #bitstar rrtstar fmtstar + ompl_sol = ompl_planner(self.gridMap, 3, current_pose, 0, goal_pose, 0, "fmtstar", False, False) + path_list = ompl_sol.omplRunOnce(10) + #print path_list + if path_list : + self.control.restart() + self.control.setPath(path_list) + tmpmap = self.gridMap.getMap() + lx = self.gridMap.size_x_ + ly = self.gridMap.size_y_ + lz = self.gridMap.size_z_ + + x = [] + y = [] + z = [] + d = 0 + for i in range(lx): + for j in range(ly): + for k in range(lz): + if tmpmap[i,j,k] < -5: + tmp = self.gridMap.mapToWorld(i,j,k) + d = d+1 + if ( d ==10 ): + x.append(tmp[0]) + y.append(tmp[1]) + z.append(tmp[2]) + d = 0 + + x2 = path_list[0] + y2 = path_list[1] + z2 = path_list[2] + ax = plt.subplot(111, projection='3d') + ax.scatter(x, y, z, c='b') + ax.scatter(x2, y2, z2, c='r') + plt.show() + return True + else: + return False + + def path_following_3d(self, current_pose): + if self.getTarget: + print "get Target" + print self.control.getPath() + + data = self.control.control(current_pose) + self.getTarget = self.control.isGetTarget() + return [self.getTarget,data] diff --git a/exercises/ardrone_navigation/navigation/ompl_planner/Point2DPlanning.py b/exercises/ardrone_navigation/navigation/ompl_planner/Point2DPlanning.py new file mode 100755 index 000000000..083827ba6 --- /dev/null +++ b/exercises/ardrone_navigation/navigation/ompl_planner/Point2DPlanning.py @@ -0,0 +1,161 @@ +#!/usr/bin/env python + +###################################################################### +# Software License Agreement (BSD License) +# +# Copyright (c) 2010, Rice University +# All rights reserved. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions +# are met: +# +# * Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +# * Redistributions in binary form must reproduce the above +# copyright notice, this list of conditions and the following +# disclaimer in the documentation and/or other materials provided +# with the distribution. +# * Neither the name of the Rice University nor the names of its +# contributors may be used to endorse or promote products derived +# from this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS +# FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE +# COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, +# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +# BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES +# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT +# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN +# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE +# POSSIBILITY OF SUCH DAMAGE. +###################################################################### + +# Author: Ioan Sucan, Mark Moll + + +from ompl import util as ou +from ompl import base as ob +from ompl import geometric as og +from os.path import abspath, dirname, join +import sys +from functools import partial + +class Plane2DEnvironment: + def __init__(self, ppm_file): + self.ppm_ = ou.PPM() + self.ppm_.loadFile(ppm_file) + self.space = ob.RealVectorStateSpace() + self.space.addDimension(0.0, self.ppm_.getWidth()) + self.space.addDimension(0.0, self.ppm_.getHeight()) + self.maxWidth_ = self.ppm_.getWidth() - 1 + self.maxHeight_ = self.ppm_.getHeight() - 1 + + self.si = ob.SpaceInformation(self.space) + + # set state validity checking for this space + self.si.setStateValidityChecker(ob.StateValidityCheckerFn( + partial(Plane2DEnvironment.isStateValid, self))) + self.si.setup() + self.si.setStateValidityCheckingResolution(1.0 / self.space.getMaximumExtent()) + #self.ss_.setPlanner(og.RRTConnect(self.ss_.getSpaceInformation())) + + def plan(self, start_row, start_col, goal_row, goal_col,plannerType,runtime): + if not self.si: + return False + start = ob.State(self.space) + start()[0] = start_row + start()[1] = start_col + goal = ob.State(self.space) + goal()[0] = goal_row + goal()[1] = goal_col + # Create a problem instance + + self.pdef = ob.ProblemDefinition(self.si) + # Set the start and goal states + self.pdef.setStartAndGoalStates(start, goal) + + self.planner = self.allocatePlanner(self.si, plannerType) + self.planner.setProblemDefinition(self.pdef) + # perform setup steps for the planner + self.planner.setup() + # print the settings for this space + print(self.si.settings()) + # generate a few solutions; all will be added to the goal + for i in range(5): + self.solved = self.planner.solve(runtime) + p = self.pdef.getSolutionPath() + print("Found solution:\n%s" % p) + ns = self.planner.getProblemDefinition().getSolutionCount() + print("Found %d solutions" % ns) + + if self.solved: + # get the goal representation from the problem definition (not the same as the goal state) + # and inquire about the found path + path = self.pdef.getSolutionPath() + print("Found solution:\n%s" % path) + return True + else: + print("No solution found") + return False + + + def allocatePlanner(self,si,plannerType): + if plannerType.lower() == "bitstar": + return og.BITstar(si) + elif plannerType.lower() == "fmtstar": + return og.FMT(si) + elif plannerType.lower() == "informedrrtstar": + return og.InformedRRTstar(si) + elif plannerType.lower() == "prmstar": + return og.PRMstar(si) + elif plannerType.lower() == "rrtstar": + return og.RRTstar(si) + elif plannerType.lower() == "sorrtstar": + return og.SORRTstar(si) + else: + OMPL_ERROR("Planner-type is not implemented in allocation function."); + + + def recordSolution(self): + if not self.si: + return + p = self.pdef.getSolutionPath() + print("Found solution:\n%s" % p) + p.interpolate() + for i in range(p.getStateCount()): + w = min(self.maxWidth_, int(p.getState(i)[0])) + h = min(self.maxHeight_, int(p.getState(i)[1])) + c = self.ppm_.getPixel(h, w) + c.red = 255 + c.green = 0 + c.blue = 0 + + def getPath(self): + # if not self.si: + # return + p = self.pdef.getSolutionPath() + p.interpolate() + pathlist = [[] for i in range(2)] + for i in range(p.getStateCount()): + w = min(self.maxWidth_, int(p.getState(i)[0])) + h = min(self.maxHeight_, int(p.getState(i)[1])) + pathlist[0].append(w) + pathlist[1].append(h) + return pathlist + + def save(self, filename): + if not self.si: + return + self.ppm_.saveFile(filename) + + def isStateValid(self, state): + w = min(int(state[0]), self.maxWidth_) + h = min(int(state[1]), self.maxHeight_) + + c = self.ppm_.getPixel(h, w) + return c.red > 127 and c.green > 127 and c.blue > 127 + diff --git a/exercises/ardrone_navigation/navigation/ompl_planner/RigidBodyPlanning.py b/exercises/ardrone_navigation/navigation/ompl_planner/RigidBodyPlanning.py new file mode 100755 index 000000000..b0233e516 --- /dev/null +++ b/exercises/ardrone_navigation/navigation/ompl_planner/RigidBodyPlanning.py @@ -0,0 +1,308 @@ +#!/usr/bin/env python + +###################################################################### +# Software License Agreement (BSD License) +# +# Copyright (c) 2010, Rice University +# All rights reserved. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions +# are met: +# +# * Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +# * Redistributions in binary form must reproduce the above +# copyright notice, this list of conditions and the following +# disclaimer in the documentation and/or other materials provided +# with the distribution. +# * Neither the name of the Rice University nor the names of its +# contributors may be used to endorse or promote products derived +# from this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS +# FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE +# COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, +# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +# BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; +# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT +# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN +# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE +# POSSIBILITY OF SUCH DAMAGE. +###################################################################### + +# Author: Mark Moll + +from math import sin, cos +from functools import partial + +from ompl import util as ou +from ompl import base as ob +from ompl import control as oc +from ompl import geometric as og + +from os.path import abspath, dirname, join +import numpy as np +import math + +import matplotlib.pyplot as plt +## @cond IGNORE +# a decomposition is only needed for SyclopRRT and SyclopEST +class RigidBodyPlanning: + def __init__(self, costMap, dimension, start, start_yaw, goal, goal_yaw, plannerType): + self.costMap = costMap + self.dimension = dimension + self.start_x = start[0] + self.start_y = start[1] + self.start_yaw = start_yaw + self.goal_x = goal[0] + self.goal_y = goal[1] + self.goal_yaw = goal_yaw + if self.dimension == 3: + self.start_z = start[2] + self.goal_z = goal[2] + self.plannerType = plannerType + self.simpleSetupControl() + + + def isStateValid(self, state): + # perform collision checking or check if other constraints are + # satisfied + if not self.costMap: + wx = state.getX() + wy = state.getY() + tmp = True + if wy < -4 and wy > 5: + tmp = False + elif wy < -0.5: + if wx > 12 or wx < 5.5: + tmp = False + return tmp + + if self.dimension == 2: + wx = state.getX() + wy = state.getY() + + mPoint = self.costMap.worldToMapEnforceBounds(wx, wy) + cost = self.costMap.getCost(mPoint[0],mPoint[1]) + if cost < -5: + tmp = False + else: + tmp = True + + elif self.dimension == 3: + wx = state.getX() + wy = state.getY() + wz = state.getZ() + mPoint = self.costMap.worldToMapEnforceBounds(wx, wy, wz) + cost = self.costMap.getCost(mPoint[0],mPoint[1],mPoint[2]) + if cost < -5: + tmp = False + else: + tmp = True + return tmp + + def simpleSetupControl(self): + if self.dimension == 2: + self.setSpace_2d() + self.setProblem_2d() + self.setPlanner_2d() + elif self.dimension == 3: + self.setSpace_3d() + self.setProblem_3d() + self.setPlanner_3d() + + def setSpace_2d(self): + # construct the state space we are planning in + self.space = ob.SE2StateSpace() + + # set the bounds for the R^2 part of SE(2) + self.bounds = ob.RealVectorBounds(2) + if not self.costMap: + self.bounds.setLow(-8) + self.bounds.setHigh(20) + else: + ox = self.costMap.getOriginX() + oy = self.costMap.getOriginY() + size_x = self.costMap.getSizeInMetersX() + size_y = self.costMap.getSizeInMetersY() + low = min(ox, oy) + high = max(ox+size_x, oy+size_y) + print ("low",low) + print ("high",high) + self.bounds.setLow(low) + self.bounds.setHigh(high) + self.space.setBounds(self.bounds) + + # define a simple setup class + self.ss = og.SimpleSetup(self.space) + self.ss.setStateValidityChecker(ob.StateValidityCheckerFn(self.isStateValid)) + + def setProblem_2d(self): + # create a start state + start = ob.State(self.space) + start().setX(self.start_x) + start().setY(self.start_y) + start().setYaw(self.start_yaw) + + # create a goal state + goal = ob.State(self.space) + goal().setX(self.goal_x) + goal().setY(self.goal_y) + goal().setYaw(self.goal_yaw) + + # set the start and goal states + self.ss.setStartAndGoalStates(start, goal, 0.05) + + def setPlanner_2d(self): + self.si = self.ss.getSpaceInformation() + if self.plannerType.lower() == "bitstar": + planner = og.BITstar(self.si) + elif self.plannerType.lower() == "fmtstar": + planner = og.FMT(self.si) + elif self.plannerType.lower() == "informedrrtstar": + planner = og.InformedRRTstar(self.si) + elif self.plannerType.lower() == "prmstar": + planner = og.PRMstar(self.si) + elif self.plannerType.lower() == "rrtstar": + planner = og.RRTstar(self.si) + elif self.plannerType.lower() == "sorrtstar": + planner = og.SORRTstar(self.si) + else: + OMPL_ERROR("Planner-type is not implemented in allocation function.") + planner = og.RRTstar(self.si) + self.ss.setPlanner(planner) + + def setSpace_3d(self): + # construct the state space we are planning in + self.space = ob.SE3StateSpace() + + # set the bounds for the R^2 part of SE(2) + self.bounds = ob.RealVectorBounds(3) + if not self.costMap: + self.bounds.setLow(0) + self.bounds.setHigh(20) + else: + ox = self.costMap.getOriginX() + oy = self.costMap.getOriginY() + oz = self.costMap.getOriginZ() + size_x = self.costMap.getSizeInMetersX() + size_y = self.costMap.getSizeInMetersY() + size_z = self.costMap.getSizeInMetersZ() + print ('o',ox,oy,oz) + print ('size',size_x,size_y,size_z) + low = min(ox, oy, oz) + high = max(ox+size_x, oy+size_y, oz+size_z) + print ("low",low) + print ("high",high) + self.bounds.setLow(0,ox) + self.bounds.setLow(1,oy) + self.bounds.setLow(2,oz) + self.bounds.setHigh(0,ox+size_x) + self.bounds.setHigh(1,oy+size_y) + self.bounds.setHigh(2,oz+size_z) + self.space.setBounds(self.bounds) + + # define a simple setup class + self.ss = og.SimpleSetup(self.space) + self.ss.setStateValidityChecker(ob.StateValidityCheckerFn(self.isStateValid)) + + def setProblem_3d(self): + # create a start state + start = ob.State(self.space) + start().setX(self.start_x) + start().setY(self.start_y) + start().setZ(self.start_z) + #start().setYaw(self.start_yaw) + start().rotation().setIdentity() + + # create a goal state + goal = ob.State(self.space) + goal().setX(self.goal_x) + goal().setY(self.goal_y) + goal().setZ(self.goal_z) + #goal().setYaw(self.goal_yaw) + goal().rotation().setIdentity() + + # set the start and goal states + self.ss.setStartAndGoalStates(start, goal, 0.05) + + def setPlanner_3d(self): + self.si = self.ss.getSpaceInformation() + if self.plannerType.lower() == "bitstar": + planner = og.BITstar(self.si) + elif self.plannerType.lower() == "fmtstar": + planner = og.FMT(self.si) + elif self.plannerType.lower() == "informedrrtstar": + planner = og.InformedRRTstar(self.si) + elif self.plannerType.lower() == "prmstar": + planner = og.PRMstar(self.si) + elif self.plannerType.lower() == "rrtstar": + planner = og.RRTstar(self.si) + elif self.plannerType.lower() == "sorrtstar": + planner = og.SORRTstar(self.si) + else: + print("Planner-type is not implemented in allocation function.") + planner = og.RRTstar(self.si) + self.ss.setPlanner(planner) + + def solve(self, runtime = None ): + if not runtime: + if self.dimension == 2: + runtime = 1 + elif self.dimension == 3: + runtime = 5 + # attempt to solve the problem + solved = self.ss.solve(runtime) + + if solved: + self.ss.simplifySolution() + # print the path to screen + p = self.ss.getSolutionPath() + print("Found solution:\n%s" % self.ss.getSolutionPath().printAsMatrix()) + #p.interpolate() + if self.dimension == 2: + pathlist = [[] for i in range(3)] + + for i in range(p.getStateCount()): + x = p.getState(i).getX() + y = p.getState(i).getY() + yaw = p.getState(i).getYaw() + pathlist[0].append(x) + pathlist[1].append(y) + pathlist[2].append(yaw) + return pathlist + elif self.dimension == 3: + pathlist = [[] for i in range(3)] + for i in range(p.getStateCount()): + x = p.getState(i).getX() + y = p.getState(i).getY() + z = p.getState(i).getZ() + pathlist[0].append(x) + pathlist[1].append(y) + pathlist[2].append(z) + return pathlist + else: + print ("can't find path") + return 0 + + +if __name__ == "__main__": + + planner = RigidBodyPlanning(None, 3, [ 12.40232229232788, 4.982222080230713, 6.667441368103027],0,[ 10, 10, 1],0, 'rrtstar') + pathlist = planner.solve() + + plt.figure(1) + plt.plot(pathlist[0], pathlist[1]) + num = len(pathlist[0]) + for i in range(num): + x1 = pathlist[0][i] + y1 = pathlist[1][i] + yaw = pathlist[2][i] + x2 = x1 + 0.5*cos(yaw) + y2 = y1 + 0.5*sin(yaw) + plt.plot([x1,x2],[y1,y2]) + plt.show() \ No newline at end of file diff --git a/exercises/ardrone_navigation/navigation/ompl_planner/RigidBodyPlanningWithControls.py b/exercises/ardrone_navigation/navigation/ompl_planner/RigidBodyPlanningWithControls.py new file mode 100755 index 000000000..926609b97 --- /dev/null +++ b/exercises/ardrone_navigation/navigation/ompl_planner/RigidBodyPlanningWithControls.py @@ -0,0 +1,256 @@ +#!/usr/bin/env python + +###################################################################### +# Software License Agreement (BSD License) +# +# Copyright (c) 2010, Rice University +# All rights reserved. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions +# are met: +# +# * Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +# * Redistributions in binary form must reproduce the above +# copyright notice, this list of conditions and the following +# disclaimer in the documentation and/or other materials provided +# with the distribution. +# * Neither the name of the Rice University nor the names of its +# contributors may be used to endorse or promote products derived +# from this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS +# FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE +# COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, +# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +# BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; +# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT +# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN +# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE +# POSSIBILITY OF SUCH DAMAGE. +###################################################################### + +# Author: Mark Moll + +from math import sin, cos +from functools import partial + +from ompl import util as ou +from ompl import base as ob +from ompl import control as oc +from ompl import geometric as og + +from os.path import abspath, dirname, join +import numpy as np +import math + +import matplotlib.pyplot as plt +## @cond IGNORE +# a decomposition is only needed for SyclopRRT and SyclopEST +class RigidBodyPlanningWithControls: + def __init__(self, costMap, dimension, start, start_yaw, goal, goal_yaw, plannerType): + self.costMap = costMap + self.start_x = start[0] + self.start_y = start[1] + self.start_yaw = start_yaw + self.goal_x = goal[0] + self.goal_y = goal[1] + self.goal_yaw = goal_yaw + self.plannerType = plannerType + self.simpleSetupControl() + + def propagate(self, start, control, duration, state): + state.setX( start.getX() + control[0] * duration * math.cos(start.getYaw()) ) + state.setY( start.getY() + control[0] * duration * math.sin(start.getYaw()) ) + state.setYaw(start.getYaw() + control[1] * duration) + + def isStateValid(self, spaceInformation, state): + # perform collision checking or check if other constraints are + # satisfied + wx = state.getX() + wy = state.getY() + if not self.costMap: + tmp = True + if wy < -4 and wy > 5: + tmp = False + elif wy < -0.5: + if wx > 12 or wx < 5.5: + tmp = False + else: + mPoint = self.costMap.worldToMap(wx, wy) + cost = self.costMap.getCost(mPoint[0],mPoint[1]) + if cost < -5: + tmp = False + else: + tmp = True + return spaceInformation.satisfiesBounds(state) and tmp + + def simpleSetupControl(self): + self.setSpace() + self.setProblem() + self.setPlanner() + + def setSpace(self): + # construct the state space we are planning in + self.space = ob.SE2StateSpace() + # set the bounds for the R^2 part of SE(2) + self.bounds = ob.RealVectorBounds(2) + if not self.costMap: + self.bounds.setLow(-8) + self.bounds.setHigh(20) + else: + ox = self.costMap.getOriginX() + oy = self.costMap.getOriginY() + size_x = self.costMap.getSizeInMetersX() + size_y = self.costMap.getSizeInMetersY() + low = min(ox, oy) + high = max(ox+size_x, oy+size_y) + print ("low",low) + print ("high",high) + #self.bounds.setLow(low) + #self.bounds.setHigh(high) + self.bounds.setLow(0,ox) + self.bounds.setHigh(0,ox+size_x) + self.bounds.setLow(1,oy) + self.bounds.setHigh(1,oy+size_y) + + self.space.setBounds(self.bounds) + + # create a control space + self.cspace = oc.RealVectorControlSpace(self.space, 2) + # set the bounds for the control space + cbounds = ob.RealVectorBounds(2) + cbounds.setLow(0,-1.5) + cbounds.setHigh(0,1.5) + cbounds.setLow(1,-3) + cbounds.setHigh(1,3) + self.cspace.setBounds(cbounds) + + # define a simple setup class + self.ss = oc.SimpleSetup(self.cspace) + self.ss.setStateValidityChecker(ob.StateValidityCheckerFn(partial(self.isStateValid, self.ss.getSpaceInformation()))) + self.ss.setStatePropagator(oc.StatePropagatorFn(self.propagate)) + + def setProblem(self): + # create a start state + start = ob.State(self.space) + start().setX(self.start_x) + start().setY(self.start_y) + start().setYaw(self.start_yaw) + # create a goal state + goal = ob.State(self.space) + goal().setX(self.goal_x) + goal().setY(self.goal_y) + goal().setYaw(self.goal_yaw) + + # set the start and goal states + self.ss.setStartAndGoalStates(start, goal, 0.05) + + def setPlanner(self): + # (optionally) set planner + self.si = self.ss.getSpaceInformation() + + if self.plannerType.lower() == "rrt": + planner = oc.RRT(self.si) + elif self.plannerType.lower() == "est": + planner = oc.EST(self.si) + elif self.plannerType.lower() == "kpiece1": + planner = oc.KPIECE1(self.si) + elif self.plannerType.lower() == "syclopest": + # SyclopEST and SyclopRRT require a decomposition to guide the search + decomp = MyDecomposition(32, self.bounds) + planner = oc.SyclopEST(self.si, decomp) + elif self.plannerType.lower() == "sycloprrt": + # SyclopEST and SyclopRRT require a decomposition to guide the search + decomp = MyDecomposition(32, self.bounds) + planner = oc.SyclopRRT(self.si, decomp) + else: + OMPL_ERROR("Planner-type is not implemented in ompl control function.") + decomp = MyDecomposition(32, self.bounds) + planner = oc.SyclopEST(self.si, decomp) + + self.ss.setPlanner(planner) + # (optionally) set propagation step size + self.si.setPropagationStepSize(.05) + + def solve(self, runtime=None): + if not runtime: + runtime = 10 + # attempt to solve the problem + solved = self.ss.solve(runtime) + + if solved: + # print the path to screen + p = self.ss.getSolutionPath() + print("Found solution:\n%s" % self.ss.getSolutionPath().printAsMatrix()) + #p.interpolate() + pathlist = [[] for i in range(6)] + #print p.getControlCount() + for i in range(p.getControlCount()): + x = p.getState(i+1).getX() + y = p.getState(i+1).getY() + yaw = p.getState(i+1).getYaw() + pathlist[0].append(x) + pathlist[1].append(y) + pathlist[2].append(yaw) + # print ('x = %f, y = %f, yaw = %f ' %(x,y,yaw)) + v = p.getControl(i)[0] + w = p.getControl(i)[1] + t = p.getControlDuration(i) + pathlist[3].append(v) + pathlist[4].append(w) + pathlist[5].append(t) + # print ('v = %f, w = %f, t = %f ' %(v,w,x)) + return pathlist + else: + print ("can't find path") + return False + + def updateMap(self, costMap): + self.costMap = costMap + + def updatePose(self,start_x, start_y, start_yaw, goal_x, goal_y, goal_yaw): + self.start_x = start_x + self.start_y = start_y + self.start_yaw = start_yaw + self.goal_x = goal_x + self.goal_y = goal_y + self.goal_yaw = goal_yaw + + def updatePlannerType(self, plannerType): + self.plannerType = plannerType + +class MyDecomposition(oc.GridDecomposition): + def __init__(self, length, bounds): + super(MyDecomposition, self).__init__(length, 2, bounds) + def project(self, s, coord): + coord[0] = s.getX() + coord[1] = s.getY() + def sampleFullState(self, sampler, coord, s): + sampler.sampleUniform(s) + s.setXY(coord[0], coord[1]) +## @endcond + + +if __name__ == "__main__": + + + planner = RigidBodyPlanningWithControls(None, 6,3,0,7.25,-3,0, 'syclopest') + + pathlist = planner.solve() + print pathlist + plt.figure(1) + plt.plot(pathlist[0], pathlist[1]) + num = len(pathlist[0]) + for i in range(num): + x1 = pathlist[0][i] + y1 = pathlist[1][i] + yaw = pathlist[2][i] + x2 = x1 + 0.5*cos(yaw) + y2 = y1 + 0.5*sin(yaw) + plt.plot([x1,x2],[y1,y2]) + plt.show() \ No newline at end of file diff --git a/exercises/ardrone_navigation/navigation/ompl_planner/RigidBodyPlanningWithODESolverAndControls.py b/exercises/ardrone_navigation/navigation/ompl_planner/RigidBodyPlanningWithODESolverAndControls.py new file mode 100755 index 000000000..a8e1b631f --- /dev/null +++ b/exercises/ardrone_navigation/navigation/ompl_planner/RigidBodyPlanningWithODESolverAndControls.py @@ -0,0 +1,257 @@ +#!/usr/bin/env python + +###################################################################### +# Software License Agreement (BSD License) +# +# Copyright (c) 2010, Rice University +# All rights reserved. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions +# are met: +# +# * Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +# * Redistributions in binary form must reproduce the above +# copyright notice, this list of conditions and the following +# disclaimer in the documentation and/or other materials provided +# with the distribution. +# * Neither the name of the Rice University nor the names of its +# contributors may be used to endorse or promote products derived +# from this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS +# FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE +# COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, +# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +# BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; +# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT +# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN +# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE +# POSSIBILITY OF SUCH DAMAGE. +###################################################################### + +# Author: Mark Moll + +from math import sin, cos, tan +from functools import partial + +from ompl import util as ou +from ompl import base as ob +from ompl import control as oc +from ompl import geometric as og + +from os.path import abspath, dirname, join +import numpy as np +import math + +import matplotlib.pyplot as plt +## @cond IGNORE +# a decomposition is only needed for SyclopRRT and SyclopEST +class RigidBodyPlanningWithODESolverAndControls: + def __init__(self, costMap, dimension, start, start_yaw, goal, goal_yaw, plannerType): + self.costMap = costMap + self.start_x = start[0] + self.start_y = start[1] + self.start_yaw = start_yaw + self.goal_x = goal[0] + self.goal_y = goal[1] + self.goal_yaw = goal_yaw + self.plannerType = plannerType + self.simpleSetupControl() + + def kinematicCarODE(self, q, u, qdot): + theta = q[2] + carLength = 5 + qdot[0] = u[0] * cos(theta) + qdot[1] = u[0] * sin(theta) + qdot[2] = u[0] * tan(u[1]) / carLength + + def isStateValid(self, spaceInformation, state): + # perform collision checking or check if other constraints are + # satisfied + wx = state.getX() + wy = state.getY() + if not self.costMap: + tmp = True + if wy < -4 and wy > 5: + tmp = False + elif wy < -0.5: + if wx > 12 or wx < 5.5: + tmp = False + else: + mPoint = self.costMap.worldToMap(wx, wy) + cost = self.costMap.getCost(mPoint[0],mPoint[1]) + if cost >= 1: + tmp = False + else: + tmp = True + return spaceInformation.satisfiesBounds(state) and tmp + + def simpleSetupControl(self): + self.setSpace() + self.setProblem() + self.setPlanner() + + def setSpace(self): + self.space = ob.SE2StateSpace() + + # set the bounds for the R^2 part of SE(2) + self.bounds = ob.RealVectorBounds(2) + if not self.costMap: + self.bounds.setLow(-8) + self.bounds.setHigh(20) + else: + ox = self.costMap.getOriginX() + oy = self.costMap.getOriginY() + size_x = self.costMap.getSizeInMetersX() + size_y = self.costMap.getSizeInMetersY() + low = min(ox, oy) + high = max(ox+size_x, oy+size_y) + print ("low",low) + print ("high",high) + self.bounds.setLow(low) + self.bounds.setHigh(high) + + self.space.setBounds(self.bounds) + + # create a control space + self.cspace = oc.RealVectorControlSpace(self.space, 2) + + # set the bounds for the control space + cbounds = ob.RealVectorBounds(2) + cbounds.setLow(0,-1.5) + cbounds.setHigh(0,1.5) + cbounds.setLow(1,-3) + cbounds.setHigh(1,3) + self.cspace.setBounds(cbounds) + + # define a simple setup class + self.ss = oc.SimpleSetup(self.cspace) + validityChecker = ob.StateValidityCheckerFn(partial(self.isStateValid, self.ss.getSpaceInformation())) + self.ss.setStateValidityChecker(validityChecker) + ode = oc.ODE(self.kinematicCarODE) + odeSolver = oc.ODEBasicSolver(self.ss.getSpaceInformation(), ode) + propagator = oc.ODESolver.getStatePropagator(odeSolver) + self.ss.setStatePropagator(propagator) + + def setProblem(self): + # create a start state + start = ob.State(self.space) + start().setX(self.start_x) + start().setY(self.start_y) + start().setYaw(self.start_yaw) + # create a goal state + goal = ob.State(self.space) + goal().setX(self.goal_x) + goal().setY(self.goal_y) + goal().setYaw(self.goal_yaw) + + # set the start and goal states + self.ss.setStartAndGoalStates(start, goal, 0.05) + + def setPlanner(self): + # (optionally) set planner + self.si = self.ss.getSpaceInformation() + + if self.plannerType.lower() == "rrt": + planner = oc.RRT(self.si) + elif self.plannerType.lower() == "est": + planner = oc.EST(self.si) + elif self.plannerType.lower() == "kpiece1": + planner = oc.KPIECE1(self.si) + elif self.plannerType.lower() == "syclopest": + # SyclopEST and SyclopRRT require a decomposition to guide the search + decomp = MyDecomposition(32, self.bounds) + planner = oc.SyclopEST(self.si, decomp) + elif self.plannerType.lower() == "sycloprrt": + # SyclopEST and SyclopRRT require a decomposition to guide the search + decomp = MyDecomposition(32, self.bounds) + planner = oc.SyclopRRT(self.si, decomp) + else: + OMPL_ERROR("Planner-type is not implemented in ompl control function.") + decomp = MyDecomposition(32, self.bounds) + planner = oc.SyclopEST(self.si, decomp) + + self.ss.setPlanner(planner) + # (optionally) set propagation step size + self.si.setPropagationStepSize(.05) + + def solve(self, runtime=None): + if not runtime: + runtime = 100 + # attempt to solve the problem + solved = self.ss.solve(runtime) + + if solved: + # print the path to screen + p = self.ss.getSolutionPath() + print("Found solution:\n%s" % self.ss.getSolutionPath().printAsMatrix()) + #p.interpolate() + pathlist = [[] for i in range(6)] + #print p.getControlCount() + for i in range(p.getControlCount()): + x = p.getState(i+1).getX() + y = p.getState(i+1).getY() + yaw = p.getState(i+1).getYaw() + pathlist[0].append(x) + pathlist[1].append(y) + pathlist[2].append(yaw) + # print ('x = %f, y = %f, yaw = %f ' %(x,y,yaw)) + v = p.getControl(i)[0] + w = p.getControl(i)[1] + t = p.getControlDuration(i) + pathlist[3].append(v) + pathlist[4].append(w) + pathlist[5].append(t) + # print ('v = %f, w = %f, t = %f ' %(v,w,x)) + return pathlist + else: + print ("can't find path") + return 0 + + def updateMap(self, costMap): + self.costMap = costMap + + def updatePose(self,start_x, start_y, start_yaw, goal_x, goal_y, goal_yaw): + self.start_x = start_x + self.start_y = start_y + self.start_yaw = start_yaw + self.goal_x = goal_x + self.goal_y = goal_y + self.goal_yaw = goal_yaw + + def updatePlannerType(self, plannerType): + self.plannerType = plannerType + + +class MyDecomposition(oc.GridDecomposition): + def __init__(self, length, bounds): + super(MyDecomposition, self).__init__(length, 2, bounds) + def project(self, s, coord): + coord[0] = s.getX() + coord[1] = s.getY() + def sampleFullState(self, sampler, coord, s): + sampler.sampleUniform(s) + s.setXY(coord[0], coord[1]) +## @endcond + +if __name__ == "__main__": + + planner = RigidBodyPlanningWithODESolverAndControls(None, 6,3,0,7.25,-3,0, 'kpiece1') + pathlist = planner.solve() + + plt.figure(1) + plt.plot(pathlist[0], pathlist[1]) + num = len(pathlist[0]) + for i in range(num): + x1 = pathlist[0][i] + y1 = pathlist[1][i] + yaw = pathlist[2][i] + x2 = x1 + 0.5*cos(yaw) + y2 = y1 + 0.5*sin(yaw) + plt.plot([x1,x2],[y1,y2]) + plt.show() diff --git a/exercises/ardrone_navigation/navigation/ompl_planner/__init__.py b/exercises/ardrone_navigation/navigation/ompl_planner/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/exercises/ardrone_navigation/navigation/ompl_planner/ompl_planner.py b/exercises/ardrone_navigation/navigation/ompl_planner/ompl_planner.py new file mode 100644 index 000000000..3294fa493 --- /dev/null +++ b/exercises/ardrone_navigation/navigation/ompl_planner/ompl_planner.py @@ -0,0 +1,69 @@ +#!/usr/bin/env python +from ompl import util as ou +from ompl import base as ob +from ompl import control as oc +from ompl import geometric as og +from os.path import abspath, dirname, join +import sys +from functools import partial + +import numpy as np +from math import sin, cos, tan +import matplotlib.pyplot as plt + +from RigidBodyPlanning import RigidBodyPlanning +from RigidBodyPlanningWithControls import RigidBodyPlanningWithControls +from RigidBodyPlanningWithODESolverAndControls import RigidBodyPlanningWithODESolverAndControls + + +class ompl_planner: + def __init__(self, costMap, dimension, start, start_yaw, goal, goal_yaw, plannerType, control = True, ODESolver = False): + self.costMap = costMap + self.dimension = dimension + + self.start = start + self.start_yaw = start_yaw + self.goal = goal + self.goal_yaw = goal_yaw + + self.plannerType = plannerType + self.control = control + self.ODESolver = ODESolver + + def setMap(self, costMap): + self.costMap = costMap + + def setPose(self,start, start_yaw, goal, goal_yaw): + self.start = start + self.start_yaw = start_yaw + self.goal = goal + self.goal_yaw = goal_yaw + + def setPlannerType(self, plannerType): + self.plannerType = plannerType + + def setControl(self, control): + self.control = control + + def setODESolver(self, ODESolver): + self.ODESolver = ODESolver + + def omplRunOnce(self, runtime=None): + if self.dimension == 3: + planner = RigidBodyPlanning(self.costMap,self.dimension, self.start, self.start_yaw, self.goal, self.goal_yaw, self.plannerType) + elif self.dimension == 2: + if self.control : + if self.ODESolver: + planner = RigidBodyPlanningWithODESolverAndControls(self.costMap, self.dimension, self.start, self.start_yaw, self.goal, self.goal_yaw, self.plannerType) + else: + planner = RigidBodyPlanningWithControls(self.costMap, self.dimension, self.start, self.start_yaw, self.goal, self.goal_yaw, self.plannerType) + else: + planner = RigidBodyPlanning(self.costMap,self.dimension, self.start, self.start_yaw, self.goal, self.goal_yaw, self.plannerType) + #costMap, dimension, start, start_yaw, goal, goal_yaw, plannerType + if not runtime: + pathlist = planner.solve() + else: + pathlist = planner.solve(runtime) + + return pathlist + \ No newline at end of file diff --git a/exercises/ardrone_navigation/navigation/path_processing/__init__.py b/exercises/ardrone_navigation/navigation/path_processing/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/exercises/ardrone_navigation/navigation/path_processing/smoothPath.py b/exercises/ardrone_navigation/navigation/path_processing/smoothPath.py new file mode 100644 index 000000000..da0407afd --- /dev/null +++ b/exercises/ardrone_navigation/navigation/path_processing/smoothPath.py @@ -0,0 +1,120 @@ +import numpy as np +import time +import math +from scipy import interpolate +import matplotlib.pyplot as plt + +class smooth: + #def __init__(self): + + def setPath(self, pathlist): + self.pathlist = pathlist + + def getPath(self): + return self.pathlist + + def inter(self,pathlist): + self.pathlist = pathlist + x = self.pathlist[0] + y = self.pathlist[1] + xnew=np.linspace(min(x),max(x),201) + #plt.figure(3) + #plt.plot(x,y,"ro") + + for kind in ["nearest","zero","slinear","quadratic","cubic"]: + #"nearest","zero" + #slinear + #"quadratic","cubic" + f=interpolate.interp1d(x,y,kind=kind) + ynew=f(xnew) + #plt.plot(xnew,ynew,label=str(kind)) + #plt.savefig("figure3.jpg") + + def Floyd(self,pathlist): + self.pathlist = pathlist + numLen = len(self.pathlist[0]) + if (numLen > 2): + + vector = [self.pathlist[0][numLen -1] - self.pathlist[0][numLen - 2], self.pathlist[1][numLen -1] - self.pathlist[1][numLen - 2],self.pathlist[2][numLen -1] - self.pathlist[2][numLen - 2]] + tempvector = [0,0,0] + #for (int i = numLen - 3; i>= 0; i--) + i = numLen - 3 + while (i >= 0): + tempvector = [self.pathlist[0][i+1] - self.pathlist[0][i], self.pathlist[1][i+1] - self.pathlist[1][i],self.pathlist[2][i+1] - self.pathlist[2][i]] + if math.fabs(math.atan2(vector[1],vector[0])-math.atan2(tempvector[1],tempvector[0])) < 0.001: + del self.pathlist[0][i+1] + del self.pathlist[1][i+1] + del self.pathlist[2][i+1] + else: + vector = tempvector + i -= 1 + numLen = len(self.pathlist[0]) + #for (int i = numLen-1; i >= 0; i--) + i = numLen-1 + while(i>=0): + j = 0 + while (j<=i-1): + #for (int j = 0; j<= i-1; j++) + + if (self.CheckCrossNoteWalkable(i,j)): + k = i-1 + while(k>=j): + #for (int k = i-1; k>=j; k--): + del self.pathlist[0][k] + del self.pathlist[1][k] + del self.pathlist[2][k] + k -= 1 + i=j + break + j += 1 + i -= 1 + + return self.pathlist + + def CheckCrossNoteWalkable(self, i ,j): + x1 = self.pathlist[0][i] + y1 = self.pathlist[1][i] + x2 = self.pathlist[0][j] + y2 = self.pathlist[1][j] + dis = math.sqrt(pow(x1-x2,2)+pow(y1-y2,2)) + return dis < 1 + + def checkYaw(self,pathlist, start): + self.pathlist = pathlist + num = len(self.pathlist[0]) + for i in range(num-1): + if (i == 0): + x0 = start[0] + y0 = start[1] + else: + x0 = self.pathlist[0][i-1] + y0 = self.pathlist[1][i-1] + x1 = self.pathlist[0][i] + y1 = self.pathlist[1][i] + x2 = self.pathlist[0][i+1] + y2 = self.pathlist[1][i+1] + vector1 = [x1-x0,y1-y0] + vector2 = [x2-x1,y2-y1] + yaw1 = math.atan2(vector1[1],vector1[0]) + yaw2 = math.atan2(vector2[1],vector2[0]) + # print ("yaw1") + # print (yaw1) + # print ("yaw2") + # print (yaw2) + yawtmp = (yaw2+yaw1)/2 + if (i==0): + yawtmp = self.setYaw(yawtmp,start[2]) + else: + yawtmp = self.setYaw(yawtmp,self.pathlist[2][i-1]) + print ("yawtmp",yawtmp) + self.pathlist[2][i] = yawtmp + return self.pathlist + + def setYaw(self, newYaw, lastYaw): + dis = newYaw - lastYaw + if math.fabs(dis) > 1.57: + if newYaw >= 0: + newYaw = -(3.14 - newYaw) + else: + newYaw = 3.147 + newYaw + return newYaw \ No newline at end of file diff --git a/exercises/ardrone_navigation/resources/__init__.py b/exercises/ardrone_navigation/resources/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/exercises/ardrone_navigation/resources/ball.png b/exercises/ardrone_navigation/resources/ball.png new file mode 100644 index 000000000..7fbb31af9 Binary files /dev/null and b/exercises/ardrone_navigation/resources/ball.png differ diff --git a/exercises/ardrone_navigation/resources/jderobot.png b/exercises/ardrone_navigation/resources/jderobot.png new file mode 100644 index 000000000..db59bb578 Binary files /dev/null and b/exercises/ardrone_navigation/resources/jderobot.png differ diff --git a/exercises/ardrone_navigation/resources/play.png b/exercises/ardrone_navigation/resources/play.png new file mode 100644 index 000000000..6c6ea8483 Binary files /dev/null and b/exercises/ardrone_navigation/resources/play.png differ diff --git a/exercises/ardrone_navigation/resources/resources.qrc b/exercises/ardrone_navigation/resources/resources.qrc new file mode 100644 index 000000000..1f733f475 --- /dev/null +++ b/exercises/ardrone_navigation/resources/resources.qrc @@ -0,0 +1,8 @@ + + + ball.png + play.png + stop.png + jderobot.png + + diff --git a/exercises/ardrone_navigation/resources/stop.png b/exercises/ardrone_navigation/resources/stop.png new file mode 100644 index 000000000..2ac1754eb Binary files /dev/null and b/exercises/ardrone_navigation/resources/stop.png differ diff --git a/exercises/ardrone_navigation/resources_rc.py b/exercises/ardrone_navigation/resources_rc.py new file mode 100644 index 000000000..4ba7e2bd9 --- /dev/null +++ b/exercises/ardrone_navigation/resources_rc.py @@ -0,0 +1,1711 @@ +# -*- coding: utf-8 -*- + +# Resource object code +# +# Created by: The Resource Compiler for PyQt5 (Qt v5.5.1) +# +# WARNING! All changes made in this file will be lost! + +from PyQt5 import QtCore + +qt_resource_data = b"\ +\x00\x00\x14\x57\ +\x89\ +\x50\x4e\x47\x0d\x0a\x1a\x0a\x00\x00\x00\x0d\x49\x48\x44\x52\x00\ +\x00\x00\x30\x00\x00\x00\x30\x08\x06\x00\x00\x01\x20\x05\xc9\x11\ +\x00\x00\x04\xc8\x69\x54\x58\x74\x58\x4d\x4c\x3a\x63\x6f\x6d\x2e\ +\x61\x64\x6f\x62\x65\x2e\x78\x6d\x70\x00\x00\x00\x00\x00\x3c\x3f\ +\x78\x70\x61\x63\x6b\x65\x74\x20\x62\x65\x67\x69\x6e\x3d\x22\xef\ +\xbb\xbf\x22\x20\x69\x64\x3d\x22\x57\x35\x4d\x30\x4d\x70\x43\x65\ +\x68\x69\x48\x7a\x72\x65\x53\x7a\x4e\x54\x63\x7a\x6b\x63\x39\x64\ +\x22\x3f\x3e\x0a\x3c\x78\x3a\x78\x6d\x70\x6d\x65\x74\x61\x20\x78\ +\x6d\x6c\x6e\x73\x3a\x78\x3d\x22\x61\x64\x6f\x62\x65\x3a\x6e\x73\ +\x3a\x6d\x65\x74\x61\x2f\x22\x20\x78\x3a\x78\x6d\x70\x74\x6b\x3d\ +\x22\x41\x64\x6f\x62\x65\x20\x58\x4d\x50\x20\x43\x6f\x72\x65\x20\ +\x34\x2e\x31\x2d\x63\x30\x33\x37\x20\x34\x36\x2e\x32\x38\x32\x36\ +\x39\x36\x2c\x20\x4d\x6f\x6e\x20\x41\x70\x72\x20\x30\x32\x20\x32\ +\x30\x30\x37\x20\x31\x38\x3a\x33\x36\x3a\x35\x36\x20\x20\x20\x20\ +\x20\x20\x20\x20\x22\x3e\x0a\x20\x3c\x72\x64\x66\x3a\x52\x44\x46\ +\x20\x78\x6d\x6c\x6e\x73\x3a\x72\x64\x66\x3d\x22\x68\x74\x74\x70\ +\x3a\x2f\x2f\x77\x77\x77\x2e\x77\x33\x2e\x6f\x72\x67\x2f\x31\x39\ +\x39\x39\x2f\x30\x32\x2f\x32\x32\x2d\x72\x64\x66\x2d\x73\x79\x6e\ +\x74\x61\x78\x2d\x6e\x73\x23\x22\x3e\x0a\x20\x20\x3c\x72\x64\x66\ +\x3a\x44\x65\x73\x63\x72\x69\x70\x74\x69\x6f\x6e\x20\x72\x64\x66\ +\x3a\x61\x62\x6f\x75\x74\x3d\x22\x22\x0a\x20\x20\x20\x20\x78\x6d\ +\x6c\x6e\x73\x3a\x78\x61\x70\x52\x69\x67\x68\x74\x73\x3d\x22\x68\ +\x74\x74\x70\x3a\x2f\x2f\x6e\x73\x2e\x61\x64\x6f\x62\x65\x2e\x63\ +\x6f\x6d\x2f\x78\x61\x70\x2f\x31\x2e\x30\x2f\x72\x69\x67\x68\x74\ +\x73\x2f\x22\x0a\x20\x20\x20\x20\x78\x6d\x6c\x6e\x73\x3a\x78\x61\ +\x70\x3d\x22\x68\x74\x74\x70\x3a\x2f\x2f\x6e\x73\x2e\x61\x64\x6f\ +\x62\x65\x2e\x63\x6f\x6d\x2f\x78\x61\x70\x2f\x31\x2e\x30\x2f\x22\ +\x0a\x20\x20\x20\x20\x78\x6d\x6c\x6e\x73\x3a\x64\x63\x3d\x22\x68\ +\x74\x74\x70\x3a\x2f\x2f\x70\x75\x72\x6c\x2e\x6f\x72\x67\x2f\x64\ +\x63\x2f\x65\x6c\x65\x6d\x65\x6e\x74\x73\x2f\x31\x2e\x31\x2f\x22\ +\x0a\x20\x20\x20\x78\x61\x70\x52\x69\x67\x68\x74\x73\x3a\x57\x65\ +\x62\x53\x74\x61\x74\x65\x6d\x65\x6e\x74\x3d\x22\x68\x74\x74\x70\ +\x3a\x2f\x2f\x62\x6c\x6f\x67\x2e\x61\x64\x64\x69\x63\x74\x65\x64\ +\x74\x6f\x63\x6f\x66\x66\x65\x65\x2e\x64\x65\x22\x0a\x20\x20\x20\ +\x78\x61\x70\x3a\x4d\x65\x74\x61\x64\x61\x74\x61\x44\x61\x74\x65\ +\x3d\x22\x32\x30\x30\x39\x2d\x30\x31\x2d\x32\x32\x54\x31\x30\x3a\ +\x30\x36\x3a\x31\x34\x2b\x30\x31\x3a\x30\x30\x22\x3e\x0a\x20\x20\ +\x20\x3c\x64\x63\x3a\x63\x72\x65\x61\x74\x6f\x72\x3e\x0a\x20\x20\ +\x20\x20\x3c\x72\x64\x66\x3a\x53\x65\x71\x3e\x0a\x20\x20\x20\x20\ +\x20\x3c\x72\x64\x66\x3a\x6c\x69\x3e\x4f\x6c\x69\x76\x65\x72\x20\ +\x54\x77\x61\x72\x64\x6f\x77\x73\x6b\x69\x3c\x2f\x72\x64\x66\x3a\ +\x6c\x69\x3e\x0a\x20\x20\x20\x20\x3c\x2f\x72\x64\x66\x3a\x53\x65\ +\x71\x3e\x0a\x20\x20\x20\x3c\x2f\x64\x63\x3a\x63\x72\x65\x61\x74\ +\x6f\x72\x3e\x0a\x20\x20\x20\x3c\x64\x63\x3a\x64\x65\x73\x63\x72\ +\x69\x70\x74\x69\x6f\x6e\x3e\x0a\x20\x20\x20\x20\x3c\x72\x64\x66\ +\x3a\x41\x6c\x74\x3e\x0a\x20\x20\x20\x20\x20\x3c\x72\x64\x66\x3a\ +\x6c\x69\x20\x78\x6d\x6c\x3a\x6c\x61\x6e\x67\x3d\x22\x78\x2d\x64\ +\x65\x66\x61\x75\x6c\x74\x22\x3e\xef\xa3\xbf\x20\x4d\x61\x64\x65\ +\x20\x6f\x6e\x20\x61\x20\x4d\x61\x63\x21\x26\x23\x78\x41\x3b\x26\ +\x23\x78\x41\x3b\x65\x78\x63\x6c\x75\x73\x69\x76\x65\x20\x66\x6f\ +\x72\x20\x53\x6d\x61\x73\x68\x69\x6e\x67\x20\x4d\x61\x67\x61\x7a\ +\x69\x6e\x65\x3c\x2f\x72\x64\x66\x3a\x6c\x69\x3e\x0a\x20\x20\x20\ +\x20\x3c\x2f\x72\x64\x66\x3a\x41\x6c\x74\x3e\x0a\x20\x20\x20\x3c\ +\x2f\x64\x63\x3a\x64\x65\x73\x63\x72\x69\x70\x74\x69\x6f\x6e\x3e\ +\x0a\x20\x20\x20\x3c\x64\x63\x3a\x73\x75\x62\x6a\x65\x63\x74\x3e\ +\x0a\x20\x20\x20\x20\x3c\x72\x64\x66\x3a\x42\x61\x67\x3e\x0a\x20\ +\x20\x20\x20\x20\x3c\x72\x64\x66\x3a\x6c\x69\x3e\x69\x63\x6f\x6e\ +\x73\x3c\x2f\x72\x64\x66\x3a\x6c\x69\x3e\x0a\x20\x20\x20\x20\x20\ +\x3c\x72\x64\x66\x3a\x6c\x69\x3e\x66\x6c\x61\x76\x6f\x75\x72\x3c\ +\x2f\x72\x64\x66\x3a\x6c\x69\x3e\x0a\x20\x20\x20\x20\x20\x3c\x72\ +\x64\x66\x3a\x6c\x69\x3e\x73\x6d\x61\x73\x68\x69\x6e\x67\x20\x6d\ +\x61\x67\x61\x7a\x69\x6e\x65\x3c\x2f\x72\x64\x66\x3a\x6c\x69\x3e\ +\x0a\x20\x20\x20\x20\x20\x3c\x72\x64\x66\x3a\x6c\x69\x3e\x61\x64\ +\x64\x69\x63\x74\x65\x64\x20\x74\x6f\x20\x63\x6f\x66\x66\x65\x65\ +\x3c\x2f\x72\x64\x66\x3a\x6c\x69\x3e\x0a\x20\x20\x20\x20\x3c\x2f\ +\x72\x64\x66\x3a\x42\x61\x67\x3e\x0a\x20\x20\x20\x3c\x2f\x64\x63\ +\x3a\x73\x75\x62\x6a\x65\x63\x74\x3e\x0a\x20\x20\x20\x3c\x64\x63\ +\x3a\x72\x69\x67\x68\x74\x73\x3e\x0a\x20\x20\x20\x20\x3c\x72\x64\ +\x66\x3a\x41\x6c\x74\x3e\x0a\x20\x20\x20\x20\x20\x3c\x72\x64\x66\ +\x3a\x6c\x69\x20\x78\x6d\x6c\x3a\x6c\x61\x6e\x67\x3d\x22\x78\x2d\ +\x64\x65\x66\x61\x75\x6c\x74\x22\x3e\xc2\xa9\x20\x32\x30\x30\x39\ +\x20\x62\x79\x20\x4f\x6c\x69\x76\x65\x72\x20\x54\x77\x61\x72\x64\ +\x6f\x77\x73\x6b\x69\x20\x3c\x2f\x72\x64\x66\x3a\x6c\x69\x3e\x0a\ +\x20\x20\x20\x20\x3c\x2f\x72\x64\x66\x3a\x41\x6c\x74\x3e\x0a\x20\ +\x20\x20\x3c\x2f\x64\x63\x3a\x72\x69\x67\x68\x74\x73\x3e\x0a\x20\ +\x20\x3c\x2f\x72\x64\x66\x3a\x44\x65\x73\x63\x72\x69\x70\x74\x69\ +\x6f\x6e\x3e\x0a\x20\x3c\x2f\x72\x64\x66\x3a\x52\x44\x46\x3e\x0a\ +\x3c\x2f\x78\x3a\x78\x6d\x70\x6d\x65\x74\x61\x3e\x0a\x3c\x3f\x78\ +\x70\x61\x63\x6b\x65\x74\x20\x65\x6e\x64\x3d\x22\x72\x22\x3f\x3e\ +\xee\x04\x51\xd2\x00\x00\x00\x19\x74\x45\x58\x74\x53\x6f\x66\x74\ +\x77\x61\x72\x65\x00\x41\x64\x6f\x62\x65\x20\x49\x6d\x61\x67\x65\ +\x52\x65\x61\x64\x79\x71\xc9\x65\x3c\x00\x00\x0f\x25\x49\x44\x41\ +\x54\x78\xda\x62\xfc\xff\xff\x3f\x03\x36\xc0\x02\x63\xd8\x75\xea\ +\xff\x67\x66\x67\x66\xf8\xfb\xf3\x2f\xc3\xa1\xf2\x8b\x8c\x8c\x20\ +\x1d\x20\x41\x5e\x49\x1e\x06\x0e\x1e\x36\x86\x1f\x5f\x7e\x31\x7c\ +\x7e\xfe\x05\xa2\xe3\xf5\xc7\x77\x0c\xbf\x45\x7e\x31\xbc\xee\xfa\ +\xc6\x20\x16\xc0\xcb\xf0\xfe\xd3\x47\x06\x26\x90\xc4\xdf\x9f\xff\ +\x18\x7e\x7e\xfc\xcd\x20\x9a\xca\xc5\xf0\x83\xed\x07\x98\xcf\x08\ +\xb3\x5c\xad\x58\x1a\x68\x07\x13\x58\xf0\x56\xef\x53\x46\x80\x00\ +\x62\xc4\xeb\x2a\xab\x56\x9d\xff\x6c\x5c\x6c\x60\x81\x2f\xb7\x58\ +\x3f\x9d\x99\x7e\x92\x9f\x25\x2c\x31\x73\x3e\xb7\x29\x27\x03\x0b\ +\x13\x0b\x03\x87\x30\x1b\x03\x2b\xfb\x37\x3e\xb0\x8e\x33\x9c\x1b\ +\x62\xf9\xbf\x72\x31\x7c\x58\xfd\x13\xe8\x12\x26\x06\xbe\x50\x56\ +\x06\x4e\x61\xe9\x03\x4c\xcf\x57\x32\xa9\xfd\xfe\xf6\x87\x41\x31\ +\x4b\x94\x41\x24\x99\x93\x01\xc4\xfe\xfe\xf6\xa9\x03\x13\x90\xb8\ +\xf7\x6a\xff\xef\xc7\xcf\x6e\xbe\x61\xf8\xf8\xe0\x1b\xc3\xdd\x49\ +\x7f\xe6\x83\x8c\x02\x08\x20\x0c\x57\x81\xec\x7c\xc0\x77\x26\x94\ +\x45\xec\x3b\xf7\x8f\xfb\xdc\x1f\xdc\x65\xfc\xdc\xda\x1b\xaa\x4f\ +\xc3\xe4\xe1\x1a\x40\x2e\x66\xe5\x60\x65\x60\xe3\x64\x61\x60\x66\ +\x63\x61\x60\x62\x66\x64\xf8\xf7\xf7\x3f\xc3\xdf\x5f\x7f\x18\x7e\ +\x7d\xff\xc3\xf0\xe5\x2a\xc7\xfb\xb3\xf3\x8e\x0b\x81\x35\x98\x16\ +\x98\x7c\xe5\x96\xff\xc7\xc5\xc1\xcb\xce\xc0\xc9\xcf\xc6\xf0\x64\ +\xd1\x7b\xb0\x21\xca\x59\xe2\x0c\x3f\x81\x61\xfd\xfd\xe3\x2f\x86\ +\x1f\x9f\x7f\x32\x70\x9d\xd0\xdb\x04\x0e\x8f\x47\x97\x9e\xbd\x17\ +\xe4\x64\xe6\x62\xff\xc9\xca\xc0\xf1\x97\x95\xe1\x35\x30\x32\x04\ +\x25\xf9\x18\x9e\x7d\x78\xce\xf0\xf3\xd3\x6f\x86\x1f\xef\x7f\x83\ +\x83\xfe\xe1\x9e\xdd\xeb\xe1\x4e\x92\x70\x97\x7a\xc6\x6f\xc0\x24\ +\xc9\xca\xc5\xcc\x00\x8a\x5e\xb8\x93\x80\xd1\xfc\xfb\xdb\x5f\x86\ +\x47\x73\xfe\xcf\x07\x06\x44\x12\x03\x48\x03\x0c\x03\x01\x07\x10\ +\xeb\xb1\xf1\x8b\x2e\x65\x17\x94\xb8\xcd\xc6\x2b\xd4\x0d\x8a\x3e\ +\x64\x35\x00\x01\x84\x33\xee\x70\x01\x16\x74\x01\xbd\x00\xdb\xe7\ +\x5c\x3a\x1f\x45\x18\xfe\x30\xfe\xfb\x79\x42\xe6\xce\xf9\x03\x5b\ +\xb5\x91\xe5\xe1\x36\x18\x39\xfa\x5c\xe6\x72\x7f\xa4\xc3\xcc\x02\ +\x74\x3f\x0b\x23\x58\xec\xdf\x1f\xa0\x1f\xfe\xfc\x65\xb0\xfe\x16\ +\xe6\xd8\xd9\x54\x73\x00\x24\x06\x4e\x80\x11\xc9\xd9\x33\xb8\x3c\ +\x1e\xeb\xb0\x71\x01\x43\x89\x97\x8d\x81\x93\x8f\x03\x8c\x41\x6c\ +\x90\xd8\x51\xee\xd5\xfb\x61\x36\x80\x35\x3c\x56\x3a\x98\xce\x0a\ +\x0c\x19\x50\x52\x63\xe7\x03\xc6\x85\x00\x1b\x18\x83\xd8\x20\x31\ +\x90\x9c\x49\xa6\xd9\x47\xb8\x06\x46\xa0\x33\x58\x58\x81\x98\x83\ +\x99\x81\x9d\x9b\x85\xe1\xc9\x82\xf7\x0c\x8f\xf6\xbc\x00\xb3\x41\ +\x62\x20\x39\x0e\x85\x9f\x7c\x70\x4f\xbf\x7a\xf5\x86\x81\xfd\x17\ +\xd0\x39\xff\xd9\x18\x38\x98\x58\x19\xde\xbc\xfc\xcc\xc0\xf8\x8a\ +\x91\xe1\xcd\xe9\x5b\x0c\x22\xf1\xc0\x7c\xf2\xfe\x17\xc3\xcf\x0f\ +\xbf\x11\x4e\x02\xf9\xfb\x3f\x30\x92\x40\x11\xf5\xef\x37\x10\xff\ +\xff\xc7\xe0\x37\xd1\x81\x41\x2a\x87\x0f\x2c\x06\x92\x83\x85\x3e\ +\x44\xc3\xef\x7f\xff\xff\xfe\xfa\xc7\xf0\xe7\xfb\x5f\x86\x5f\x9f\ +\xff\x30\x48\xa4\xf1\x30\x1c\xbf\x7c\x12\xcc\x06\x89\x81\xe4\x9e\ +\x2c\x64\x38\x0d\xd7\xf0\x74\x19\x13\xfb\x9f\x1f\xff\x18\x7e\x7f\ +\xf9\x03\x4e\x33\x3f\xde\x41\x30\x88\x0d\x12\x03\xc9\x01\x93\x85\ +\x19\x4a\x3c\x00\xb3\x93\x84\x4c\xdc\xff\xe7\x4c\xac\x4c\x28\xf1\ +\xf0\xe7\xcb\xbf\xdf\x77\xa7\x3d\x63\x83\xc7\x1c\x5a\x5a\x92\x65\ +\xe1\xe4\x8d\x63\x17\x10\x3f\xc3\x26\x20\xb6\x0f\xc8\x37\x05\x62\ +\x61\x64\x35\x00\x01\x68\xab\x96\xd0\x26\xa2\x28\x7a\xdf\x9b\x99\ +\x4c\xd2\x94\x50\xd1\x22\x28\xa8\x48\xe3\x2a\x82\x0a\x7e\x28\xb8\ +\xa9\x41\x05\x3f\xd9\x4b\xc5\x9d\x4b\x91\x22\x22\xd8\x45\x5d\x75\ +\xeb\xca\xbd\x0b\x29\xe2\x4a\x28\xe8\xce\x5d\xab\x0b\x89\x6e\xd4\ +\x85\x48\x6a\x25\x03\x16\x35\x18\x3a\x0d\xc9\xfb\x78\xcf\x7b\xd3\ +\x40\xd3\x28\x74\xe1\x83\xcb\xcc\x64\xe6\x9d\x9b\x7b\xee\x39\xf7\ +\xed\x58\x4b\x3b\x5d\xe1\xdf\x5e\x4c\xdf\xbc\xf5\xf0\x83\x7d\x73\ +\xa3\x70\x68\x7d\x8c\xa4\x24\x21\x85\x27\x28\x5b\x06\xd5\x18\xfe\ +\x73\xda\xd8\xd6\x8b\x7d\x9f\xaf\xd5\x2e\x1e\xbb\x7f\x77\x26\x1d\ +\xc4\xd9\x56\x01\x34\x5b\xa8\x36\x2a\x10\x4b\x10\x78\xbe\xe0\x3d\ +\x99\x25\x21\xd0\x67\x3d\xb8\x31\xc6\xb7\x1a\x9a\xd6\x3c\x3f\xbf\ +\xe6\xda\xd5\xf1\xda\xa9\xf9\xb9\xd9\x4f\x43\x13\x40\x8e\xf9\x83\ +\x9d\x12\x0c\x11\xe4\xa4\x53\x9c\xe4\x08\x22\x41\x02\xc9\x50\x82\ +\x90\x40\x67\x70\x68\xc9\x90\x86\x8e\x7a\x9a\x54\xcf\xcb\x01\xe6\ +\x19\x7d\x7b\x7c\xf1\xe5\xb3\xc7\x57\xfa\xb2\xc0\xaa\x5c\x3d\x9b\ +\xc4\x07\x3c\x78\xc8\xc3\x3a\x8a\x43\x8a\x46\x42\xca\x15\x23\x8a\ +\xf9\x1c\x48\x16\x5a\xf4\xf1\x49\x83\xcf\x84\x90\x8d\xc5\x7e\x29\ +\xe5\xdc\xef\x78\x8f\xef\xf0\x3d\xf6\x61\x7f\xfb\x44\xfd\xf2\x9d\ +\xd9\x07\x97\xb6\x54\x70\x74\x66\xa2\x17\x8d\x32\x31\x31\x12\x04\ +\x14\xe5\xbd\x97\xc2\x82\x7f\x5e\x79\xf4\x93\xa4\x08\x48\x08\x2f\ +\xb1\xb4\x9b\xd2\x9e\xa9\x22\x95\xca\x23\x4e\xcc\xaa\xc3\xa3\x8b\ +\x43\xf1\x08\xc3\x18\x6b\x3d\xdf\xbd\xfc\xe5\xfd\xeb\xc9\x7e\x93\ +\x7f\xff\x58\xef\xc6\x42\x86\x81\x62\x6a\x34\x83\x1a\x0e\xcb\xf7\ +\x84\x6b\x40\x69\x7b\x83\x29\x62\xaa\xb2\xa2\x2b\xe7\xcb\x94\x3f\ +\x29\x68\x25\x59\xf5\x09\x36\x8c\x4b\x82\xc0\x51\x95\x7c\x4b\x1b\ +\x5b\x54\xf4\x7d\x51\x2e\xed\xbf\x6e\xab\x42\x81\x53\xe6\x32\x00\ +\xef\xa8\x51\x38\x5f\x6a\xa3\xa9\x36\x7f\x8e\xea\x6b\xef\x5c\x63\ +\xd7\x74\x93\x74\xd3\x38\x30\xc7\x3d\xef\xc1\x3e\x67\x9e\xd4\xb2\ +\xf3\xc5\xf4\xb6\x26\xb3\xdb\x96\xd9\x6d\x67\xe0\x36\x34\x19\x21\ +\x71\x45\x93\xc3\xcd\x26\x0b\x37\x2d\x5c\x93\x55\xd6\xe4\x6e\x96\ +\x04\xf3\x20\x35\xaa\xb9\x20\x0e\xb3\x95\x57\x87\xca\x94\x93\x4c\ +\xec\x9a\xb4\x4b\xc5\xb2\x18\x77\x12\xcd\x64\x2a\x10\x03\x32\xed\ +\x4f\x24\xe5\x83\x07\xca\x2b\x06\x9e\xfa\xa7\x0f\x06\x92\xdd\x1b\ +\x3b\x6d\x6f\x17\x8f\xd0\x5e\x80\x67\xfd\xa5\xcd\x91\xa7\x3b\x56\ +\x25\x4f\x45\x9d\x1f\x2f\x30\xf0\xaf\x61\x18\xff\x7d\x54\xfc\x11\ +\x80\x18\x6b\x09\xa9\x2a\x8a\xa2\xfb\x9c\x7b\xdf\x55\x5f\xf8\x44\ +\x2c\xe8\x6b\x3f\x07\x1a\x36\x88\x4a\x28\x28\x6a\x12\x92\x43\x4b\ +\x0a\x82\x68\x50\xd0\x20\xa8\x51\x38\x09\x07\x0a\xd1\x20\xa4\x06\ +\x51\x42\x45\xbf\x41\x8d\xc4\x02\xfb\x47\xd1\x1f\xca\xfe\xcf\xfc\ +\x95\xff\x7a\xa2\xf9\xf4\x7d\xf5\xdd\xd3\xde\xe7\x9c\x7b\xbb\xef\ +\x29\x56\x83\x48\xd8\x5c\x3d\x67\x9f\xbd\xcf\x6f\xaf\xb5\x8e\xff\ +\x3c\x01\x87\x7f\xfc\x63\xce\xd4\x59\x5d\x53\xb7\xb6\xf9\xc1\x93\ +\xf3\xac\x68\x68\xbe\x35\x3b\xee\xe7\xb9\x29\xcb\x8e\xf3\x49\x3b\ +\x9c\x95\x88\x7f\xcd\x1e\x2b\xce\x2f\x6d\xbe\x7a\xee\xd4\x9e\xbf\ +\xde\xa2\xba\x63\xc7\xfd\x57\x1a\x9b\x5b\xf2\xca\xfb\x8b\x10\x83\ +\x18\xd3\xd7\x93\x7b\x90\x94\x4e\x19\x21\x09\x62\x5d\x39\x3f\x4a\ +\x58\xd9\xa5\xcb\x67\x4e\x1e\xf8\xa3\x04\x87\x8f\xd4\x6e\xba\xfb\ +\xbd\xa9\x31\xab\x30\x12\x60\x86\x21\x51\x54\xde\x7f\xcf\x2d\x92\ +\x43\x6c\x55\x0b\x84\xa8\x22\x95\x82\xd8\xed\x25\xef\x5f\xdd\xbf\ +\xbe\x72\xc6\x04\xd5\x35\xb5\xc5\x77\x42\x8d\x2f\xac\x85\x89\x5c\ +\x83\x20\x5a\xcb\x0c\xa6\x93\x38\x38\x24\x9c\x42\xb3\x6d\x5d\x03\ +\x08\x0f\x58\x0f\xb1\x5b\x85\x53\x92\xa4\x1d\x72\x73\xf7\x8d\x27\ +\x6e\x70\x82\x69\x0b\x71\x08\xcd\x87\x28\x69\x12\x5a\x66\x6b\x23\ +\xa4\xa5\x36\x4b\xf9\x48\x5f\x1c\x93\xb3\xa5\xbb\x94\x64\xcd\xb4\ +\x09\xca\xb7\xef\x6e\xf2\x2f\x1f\xcb\x37\xb8\x0e\x4e\xca\x04\x21\ +\xc2\xb4\x4c\x1d\xdc\xf8\x95\x80\x7e\xa7\x36\xea\x43\x1f\xc3\xe1\ +\x0d\x1c\xdb\x6e\xbd\xdc\x39\x6d\x82\xa1\x82\x4f\x1b\x89\x54\x98\ +\x66\x31\xc2\x1f\x12\xb7\x24\xe4\xfb\x9f\x86\x10\xbe\xd5\xac\xa5\ +\x65\x53\xbb\xea\x93\x3e\x3e\x05\x29\x34\x96\x08\x8b\x94\x74\x5a\ +\x02\x22\x07\x6b\x51\x34\xc0\x99\x72\x24\xaa\x94\x32\xcf\xa7\x58\ +\xcd\x1f\xce\x85\xd6\x86\x3e\x18\x0e\x86\xd5\xec\x69\xdb\x5c\xc6\ +\x53\xbe\x2e\xbd\x62\x8c\xe0\xf0\xfb\xf2\xb4\x04\x1f\xdb\xba\xf6\ +\xc9\xf3\xd3\xbc\x4b\xc0\x26\x79\xd8\x19\xc4\x0d\x98\x65\xf9\x21\ +\xfa\x4c\xc0\x9b\x13\x1d\x12\xfb\x1d\x7d\x23\x7d\xbc\x60\x88\x31\ +\x72\x96\xc5\x02\x69\x85\x16\x0c\x06\xe7\x04\x8a\xe2\xb8\xdc\xa4\ +\x24\x1a\x03\xcd\xc2\x2e\x13\x09\xc1\xc4\xe0\x91\xf1\x28\x24\x22\ +\x49\xc9\x68\x28\x2c\xa1\xed\xf4\x00\x84\xd9\x08\x94\xec\x5d\x0c\ +\x93\x49\xc5\x62\xf4\x36\x20\x26\x93\xcc\x66\x0a\x2b\x2d\xc1\x40\ +\x6f\x4f\x92\x47\x04\x18\x13\x0e\x9b\x71\x48\x10\x93\x69\x8b\x46\ +\x62\x90\x40\x9d\x47\xcb\x57\x7c\x00\x60\x60\x8c\xbe\xcb\x23\xe0\ +\xaf\xe0\x9a\xd1\x3c\xac\x16\xb3\x59\xe6\x21\xf7\x7b\x80\x58\x15\ +\x92\x63\x4e\x61\x51\xbb\xfa\xe0\xd6\x70\xd8\x51\x5f\x01\x81\x4a\ +\x9f\xeb\xe7\xf8\xc8\x27\x60\x04\xe7\x97\x81\x45\xb7\xd0\x63\xa7\ +\x94\xb5\x76\x26\x99\xd8\x5a\x06\xaa\x00\xe5\x87\x36\xc0\x17\x7f\ +\x07\x3c\xec\x7c\x24\xfb\x5c\xb9\x6b\x2b\xe8\x20\xbf\x58\x37\x8c\ +\x64\x26\xb8\x90\x4a\x40\x03\x37\x85\xe9\x0d\xae\xf8\x59\xfd\xbd\ +\x74\xdd\x02\x98\xbb\x35\x0f\xde\x0e\xb7\x80\x08\x83\xe2\xdf\x09\ +\xcd\x66\x9e\x24\x04\x21\xe1\x16\xf6\x21\x6d\x8b\x90\x8d\xec\xc1\ +\x6b\xfc\xb5\x54\x6b\x9a\xfe\x24\x89\x6b\x9e\x5d\x73\x70\x05\x44\ +\x57\x8e\x42\x57\x5f\xb7\x2b\x51\x26\x35\xd9\xdb\x0e\xe1\xeb\x71\ +\x98\x88\xd6\xba\x7f\x3a\xb8\xae\xb4\xe3\xa2\x93\x71\x61\x92\x23\ +\x7e\x21\xc5\x52\x72\x5b\x5b\x3b\xdb\x7e\x5d\x43\xf0\x6c\xe1\x04\ +\x4d\x24\xe5\xae\x84\xda\x7a\x2f\xb2\xe7\xf4\xb6\x9d\x52\xc9\xa4\ +\x02\xfa\xaf\xf2\x8b\xea\x59\xa1\x55\x42\x42\xc8\x2b\xa8\x6e\x46\ +\x86\xb9\x22\x4b\xa8\x95\xe0\x98\xf0\x07\x31\x40\xff\x2d\x98\x11\ +\xae\x91\xe8\xef\xa1\x3e\xda\xac\x14\x05\xd7\x10\xc0\xa4\x24\xf5\ +\xa2\xa9\x3c\x50\x57\x51\xd8\xf8\xca\x17\xa1\x91\xc7\x6c\x3d\x4e\ +\xb4\xfd\xb7\x84\x83\x49\xce\xce\xdb\x66\xef\x32\x66\x71\x9f\xac\ +\xea\x19\xf8\x80\x92\x8c\xbe\x12\x3d\x63\xef\x58\x19\x06\x1f\xfc\ +\x63\xd2\xc7\x24\xf9\xf8\xb9\x39\xaf\x4a\xac\x32\xb2\xf1\xac\xa4\ +\x6c\x74\x2a\x45\x25\x18\xff\x0c\xdf\x7e\x3c\x63\xf5\x18\xf8\xe8\ +\x5f\xab\x0a\xdc\x8e\x02\x7a\x0e\xa1\xe5\xe1\x33\xbb\x0a\x11\x6d\ +\x35\x72\xe7\x22\x9c\xfe\xa8\x10\x76\xab\x9d\x8c\xdf\x9c\x8c\x8d\ +\xb5\x60\x3f\xdd\xf9\x10\xc6\x89\xff\x17\xd9\xf2\x53\x80\x6e\xad\ +\x35\x36\x8a\x2a\x0a\x9f\x3b\xbb\xdb\xdd\xed\x76\xfb\xa4\x85\x96\ +\x16\x50\x20\x48\x78\x94\x20\x60\x43\x20\x3c\x4c\xe4\x61\x8c\x21\ +\x21\x46\x7f\xf8\xf8\xe9\xe3\x9f\x46\x25\x40\xb4\x1a\x40\xf0\x87\ +\xfe\xf2\xf1\xd3\x47\x4c\xf8\x81\x41\x63\x02\xc1\x44\x6a\x84\x28\ +\x51\x0c\xa2\x3c\x5a\xfb\x80\xb2\x55\x28\xec\x96\xb6\xb3\xdb\x2e\ +\x3b\x3b\x73\x3d\xe7\x3e\x66\x66\xcb\xb6\xd2\x20\x90\xb8\x9b\x9b\ +\xb9\x3b\x77\x67\xe7\x3b\xf7\x9e\x7b\xce\xf7\x9d\xd9\x3b\x7e\x83\ +\x7b\x4a\x8b\x26\xf3\x7a\x79\x7b\xeb\xe3\x57\x93\x03\xeb\xae\x0f\ +\x9b\x4d\x83\x43\x66\x7d\xce\xca\x45\x86\xcd\x74\x35\x8d\x95\xc7\ +\xcb\x06\x4a\x42\x25\xd9\xca\x8a\xf8\xe5\xaa\xf2\x78\xa2\x6e\x4a\ +\x75\xdb\x7b\x7b\x5a\xbf\xbe\x67\x2b\xf0\xc6\xae\xbd\xb3\xce\x74\ +\xf4\xec\x3e\xdb\xd9\xbd\x32\xbe\xc4\xac\x0e\x4d\xcd\xc6\x58\x89\ +\x13\x60\xfe\x1c\xc0\x8a\x5c\xc8\x5d\xd2\x86\x11\xcc\xb0\xad\xfe\ +\x48\xc6\x3c\x15\x1f\x58\x30\x77\xf6\x8f\x0b\xe7\xdd\xbf\xe3\xed\ +\x9d\xdb\x2e\xde\x31\x03\xb6\xb7\xee\x5e\x78\xfc\xe4\xe9\x8f\xcc\ +\xba\xc4\x82\x48\x63\xa6\xdc\x08\x63\x52\x65\x06\x14\x70\x4a\x26\ +\x35\x2e\xd3\xfd\x22\x06\xc8\x9c\xe2\x88\x8e\x9f\x7f\x42\x0e\xb5\ +\xc6\xa5\xd8\x70\xec\x6a\xd3\xd9\x55\xcb\x9a\x5f\xd8\xd3\xba\xe3\ +\xcc\x7f\x62\x00\xf1\xd4\xc3\xdf\x1f\xff\x38\xbc\x28\xd5\x10\x6a\ +\x18\x8d\x13\x85\xa4\x99\x16\xb9\x59\xf5\x05\x0b\x31\x24\x68\x11\ +\x0c\x55\xda\xf6\x82\xb5\xe3\xc5\x2f\x2e\x03\x25\xf8\x33\xa7\x88\ +\x6b\x18\x8d\x55\xd6\xcc\xf5\x85\xcd\x1b\x7f\xd4\xfc\xbd\x69\xed\ +\xaa\xe7\x75\x25\x6c\xd2\x06\x10\x7b\xff\xf6\xd8\x89\xc3\x99\x86\ +\xde\xe6\x92\x19\x99\x0a\xa6\x6b\x21\x02\xb0\xe1\x45\x6f\x11\x60\ +\x15\x2f\x56\xfa\x90\x81\x3e\x82\xa7\x17\xdd\x15\xe0\xbe\x32\x07\ +\x19\xe3\x78\xa9\x5c\x64\x5c\x32\x44\x8e\xdf\xe8\x8d\x0e\xc5\xfe\ +\x9a\x79\xfa\x91\xd5\x2d\x9b\x8a\xd5\x58\xc6\x35\x80\x34\xcd\xc1\ +\x23\x47\xf7\xd7\x6c\xbc\x36\x0b\x42\x8e\x21\x89\x3b\x73\x81\x0b\ +\x0e\x67\x28\x2e\x67\xc8\xd5\x10\x8c\x56\x67\x0b\x1d\xd4\x59\x11\ +\xf7\x51\x33\x2e\x13\xae\x9a\x75\x52\x18\x1a\xbc\x4b\x1d\xa8\xd4\ +\x82\x5f\xb4\xc0\x49\x1e\xaa\xbd\xb8\x65\xc3\xfa\x27\xfd\x35\xe2\ +\x71\x0d\xd8\xf6\xe6\xae\xd5\x07\x8f\xb4\x7d\x56\xf3\x68\xff\x4c\ +\xa1\x9b\x54\x91\xc8\x50\x35\x05\x23\x10\x50\x85\x23\xbc\x58\xf5\ +\x35\x13\x90\xf9\xd4\x70\x8d\x1a\x6b\x80\x06\x2b\x67\xbd\x90\x78\ +\x91\x3c\x72\x94\x21\x0e\xf6\xb9\xad\xfa\x24\x9d\xf2\x9c\x27\xbf\ +\xa9\xed\xdd\xb2\x61\xdd\x33\x7b\xdf\xda\x79\x6c\x5c\x03\xd0\x6d\ +\x82\x07\x0e\x7d\x77\x3a\xb2\x26\x31\xcf\x08\x43\xc0\x0f\x5e\x4b\ +\xa8\x80\x60\xd9\x52\x56\xb1\xa0\xb7\x1a\xa4\x6c\xfa\xbe\x4c\x41\ +\xea\x5a\x0a\x6a\x9b\x2b\x61\xfa\xca\xa9\x42\x37\x14\xec\x61\x87\ +\xbb\x3e\x4f\xc0\x6d\xff\xac\x2b\x09\x46\xb3\x6e\xe7\xb9\x27\xcb\ +\xb4\x11\x59\x66\x8f\xb4\x4d\xef\xd8\xba\xf9\xe1\x66\x74\xa7\x7c\ +\xd1\x3c\xd0\xde\x73\xe9\x5d\x63\x4e\xaa\x81\x22\x8c\x70\x11\xb9\ +\x4f\xc5\xcc\xbb\xe0\xb5\x26\x24\xe0\xe2\x73\xc0\x35\x2a\x5a\x12\ +\x85\xaa\xd2\x1a\x70\xba\x0c\x48\x74\x0d\xc0\x88\x95\xa1\xda\x26\ +\xcc\x5a\x53\x0f\xf1\x19\xa5\x0a\xb8\xf2\x79\xe4\x73\x2c\x40\x60\ +\x69\xa5\x6c\xe9\x5e\x74\x23\x34\x42\xf0\xe6\x3c\xb9\xab\x0d\x86\ +\xda\xf8\x0e\x62\x42\x6c\xf5\xed\x3d\xbd\xef\xe0\x99\x57\x8b\x1a\ +\x70\xbe\xa3\x73\x39\x7f\xd0\x8c\x5a\x48\x00\x59\x10\x55\x02\xa7\ +\x90\xc8\x85\x3b\x04\x18\xa9\xe0\x80\xe8\x1b\x88\xc4\xe0\x4c\x3c\ +\x1a\xe1\xe2\x4d\x9b\x1b\xd5\x81\x25\x67\x53\xdc\x11\x71\x45\x8d\ +\x98\x60\x1a\xfd\x5f\x99\x90\x70\x92\x90\xb1\x4d\xa8\x5b\x5a\x05\ +\xb5\x2b\xaa\xe5\xb3\x03\x7c\xd3\xd5\x36\xc8\xbd\xe0\x88\x12\xa3\ +\xaa\x5f\x3a\x52\x10\xeb\x3d\x61\x93\x7f\x55\x67\x4a\xcf\xff\xd2\ +\xd5\x32\x6e\x26\xbe\xd0\xdd\x15\x98\xb2\x78\x24\x68\xa4\xbd\xca\ +\x93\x76\x1f\xa9\xc4\x41\x68\x54\x23\xa4\xb8\x64\x9e\x79\xe2\x0c\ +\xcf\xa5\x87\x47\xc4\xa3\x12\x5a\x39\x4e\x91\x09\x54\x44\x52\xaf\ +\x10\x44\xe0\xfa\xf1\x51\x48\xfe\x90\x80\x91\x5c\x06\x58\x35\x87\ +\x69\xeb\xab\x20\x52\x17\x56\xac\x59\xe9\x00\xcb\x53\xf5\xba\xca\ +\xc5\x05\xcd\xe7\xc1\x6b\x88\x71\x5c\x03\x46\x32\xe9\xc1\xdc\xa8\ +\x43\x29\x35\x60\xa0\xb4\x62\x0e\x53\x33\x4d\xcd\x91\x47\x5c\x09\ +\xb1\x0a\xf4\x66\x7a\x4c\x46\x29\xcb\xb2\x70\x15\xf2\x22\x17\x30\ +\xfc\x9e\x0d\xbe\xca\x18\xf7\x0a\xea\x54\x5f\x5d\xfb\x54\x0b\xae\ +\x46\x25\xb4\xf7\xfd\x09\x03\x43\x83\x2e\x71\x2e\x7e\x94\xab\x90\ +\x1f\xe5\x0e\x61\x9c\x88\x0b\xfd\x96\x4b\xb1\x96\x28\xba\x32\x17\ +\x3b\x5c\x3d\x81\xe0\xba\x92\xcf\xd4\x46\x64\x5e\xec\xd6\x7d\x9b\ +\xc9\x1a\xb9\x8b\xd4\x4d\x01\xe2\xd5\xb4\xa8\x1e\x56\x3c\xb1\x08\ +\x06\xc3\xd7\xa1\xf3\x4a\x37\xf4\xe6\x7a\xe0\x42\xa7\x04\xe8\x6e\ +\x6e\x5f\x62\xd3\x52\xd0\x4d\x7e\x94\xac\x93\x90\x26\x8c\x13\x19\ +\xb0\x3f\xd3\x0e\x5b\xa2\x8d\xbc\x8a\xd0\x70\xda\x5f\x0e\x14\x02\ +\xd5\x09\x07\xc7\xa4\xfe\xd5\x80\xd5\xe6\x54\x59\x96\x2a\xd8\x4b\ +\x1e\x9b\x0f\xf7\xad\x6e\x84\xce\x54\x27\x24\xcd\x24\xfc\x94\x3a\ +\x21\xaf\xb5\x6f\x2e\x82\xf2\x02\x7d\xea\x19\x03\xfa\x37\x49\x20\ +\x9c\x83\x7e\xbc\xd3\x17\xff\xa6\x99\x5e\xa9\x6c\xe1\xaf\xc5\xe6\ +\x40\x1d\x1b\xb3\x0f\x58\xc0\xab\xcc\xd2\x1e\x60\x41\xdf\x79\xda\ +\x94\x27\x42\xb0\x78\xe3\x5c\xb0\x2a\x72\x90\x18\xe8\x83\x5c\xde\ +\x2a\x1e\x46\x1d\xf0\xc5\x7f\xcf\xc7\xb5\xd0\xd6\x06\xf9\x35\x71\ +\xba\x83\x93\xba\xd9\x87\xea\xe6\xfd\x5b\xd1\x64\xfb\x2a\x1f\xe2\ +\xcf\xc6\xe6\xc2\x54\xbf\x11\x32\x91\x29\xe0\x14\x5a\xfd\x95\x15\ +\x3f\xb5\x30\x3c\x4e\x54\x90\x89\xb9\xcf\x4d\xf4\x4a\xba\x86\x10\ +\x78\x90\x86\x50\xa8\x55\x89\xcc\x07\xfe\x53\x04\xff\xfa\x64\x34\ +\xdf\x8b\xf1\x85\x7c\x5b\xf9\x12\xde\xa4\x4b\x46\xfe\xd2\x91\x1f\ +\xb4\xe1\x1f\x67\x3e\x01\x7a\x93\x01\x63\xa9\x04\x17\x89\xeb\x66\ +\x63\xb8\x3b\x3e\x74\x0a\x48\xb0\xee\x45\xf0\x1f\x4e\x9a\x8d\xa2\ +\x11\x75\x78\xf8\xa4\x6a\x25\x5f\x56\x3a\x1b\x6a\xdd\xd9\x35\xc6\ +\xce\xb8\xd7\x2f\x60\xa4\xc5\xc8\x1c\xf7\xc8\x9c\xb7\x1a\x7e\x32\ +\x27\xfb\x99\x2e\x21\xe3\x4f\xe2\x55\xcf\x21\xf8\xab\xb7\xa5\x07\ +\xe8\x09\x05\x1e\x3e\xa8\x58\xce\x17\x97\x3d\x00\xd3\x34\x85\xd6\ +\x33\x3d\x31\x9d\xf6\x62\xd1\xc4\x74\x5a\x82\x4f\xb7\xc3\x95\xc1\ +\x9f\xd9\xef\x78\xe6\xa5\xb1\x35\x88\xdb\x56\x64\x68\x48\x08\x0f\ +\x4f\xd3\xac\x54\x2c\xe5\xb3\xcb\xe6\x43\x3d\x3d\xe5\x02\xe6\x73\ +\x19\xc6\x6e\x41\xd0\x70\xbf\x4b\x71\xf3\x1c\x5c\x1e\xfa\x95\x75\ +\xe3\xa7\xcf\x69\xc5\x11\xb8\x75\x57\x44\x3d\x1a\xd4\x84\x87\xad\ +\xd8\x36\x87\xa7\xf1\xc6\x48\x3d\x94\x07\x2b\x20\x12\x28\xc3\x16\ +\x85\x90\x11\x81\x90\x60\xd0\x59\xb0\x6c\x6a\x26\x64\xf3\x43\x90\ +\xcd\x5e\x81\xe1\x1b\x97\x59\x1f\x0e\x1d\xc2\x76\x40\x3f\x9c\xba\ +\x2b\x9a\x18\x5d\x84\xfe\xda\x80\x24\x07\x4a\xc9\x06\xf5\x57\x87\ +\x30\xb6\x12\xd5\x82\xaa\x19\xbe\xda\x99\xa3\x5a\x5e\xb5\x9c\x6a\ +\x54\xe8\xa4\x7a\xcc\x28\x11\x01\x6c\x99\xf1\xea\x33\xff\xdb\xb2\ +\xca\x3f\x4d\xc3\x2f\x8a\xfb\x02\xd0\xf0\x00\x00\x00\x00\x49\x45\ +\x4e\x44\xae\x42\x60\x82\ +\x00\x00\x12\xd1\ +\x89\ +\x50\x4e\x47\x0d\x0a\x1a\x0a\x00\x00\x00\x0d\x49\x48\x44\x52\x00\ +\x00\x00\x30\x00\x00\x00\x30\x08\x06\x00\x00\x01\x20\x05\xc9\x11\ +\x00\x00\x04\xc8\x69\x54\x58\x74\x58\x4d\x4c\x3a\x63\x6f\x6d\x2e\ +\x61\x64\x6f\x62\x65\x2e\x78\x6d\x70\x00\x00\x00\x00\x00\x3c\x3f\ +\x78\x70\x61\x63\x6b\x65\x74\x20\x62\x65\x67\x69\x6e\x3d\x22\xef\ +\xbb\xbf\x22\x20\x69\x64\x3d\x22\x57\x35\x4d\x30\x4d\x70\x43\x65\ +\x68\x69\x48\x7a\x72\x65\x53\x7a\x4e\x54\x63\x7a\x6b\x63\x39\x64\ +\x22\x3f\x3e\x0a\x3c\x78\x3a\x78\x6d\x70\x6d\x65\x74\x61\x20\x78\ +\x6d\x6c\x6e\x73\x3a\x78\x3d\x22\x61\x64\x6f\x62\x65\x3a\x6e\x73\ +\x3a\x6d\x65\x74\x61\x2f\x22\x20\x78\x3a\x78\x6d\x70\x74\x6b\x3d\ +\x22\x41\x64\x6f\x62\x65\x20\x58\x4d\x50\x20\x43\x6f\x72\x65\x20\ +\x34\x2e\x31\x2d\x63\x30\x33\x37\x20\x34\x36\x2e\x32\x38\x32\x36\ +\x39\x36\x2c\x20\x4d\x6f\x6e\x20\x41\x70\x72\x20\x30\x32\x20\x32\ +\x30\x30\x37\x20\x31\x38\x3a\x33\x36\x3a\x35\x36\x20\x20\x20\x20\ +\x20\x20\x20\x20\x22\x3e\x0a\x20\x3c\x72\x64\x66\x3a\x52\x44\x46\ +\x20\x78\x6d\x6c\x6e\x73\x3a\x72\x64\x66\x3d\x22\x68\x74\x74\x70\ +\x3a\x2f\x2f\x77\x77\x77\x2e\x77\x33\x2e\x6f\x72\x67\x2f\x31\x39\ +\x39\x39\x2f\x30\x32\x2f\x32\x32\x2d\x72\x64\x66\x2d\x73\x79\x6e\ +\x74\x61\x78\x2d\x6e\x73\x23\x22\x3e\x0a\x20\x20\x3c\x72\x64\x66\ +\x3a\x44\x65\x73\x63\x72\x69\x70\x74\x69\x6f\x6e\x20\x72\x64\x66\ +\x3a\x61\x62\x6f\x75\x74\x3d\x22\x22\x0a\x20\x20\x20\x20\x78\x6d\ +\x6c\x6e\x73\x3a\x78\x61\x70\x52\x69\x67\x68\x74\x73\x3d\x22\x68\ +\x74\x74\x70\x3a\x2f\x2f\x6e\x73\x2e\x61\x64\x6f\x62\x65\x2e\x63\ +\x6f\x6d\x2f\x78\x61\x70\x2f\x31\x2e\x30\x2f\x72\x69\x67\x68\x74\ +\x73\x2f\x22\x0a\x20\x20\x20\x20\x78\x6d\x6c\x6e\x73\x3a\x78\x61\ +\x70\x3d\x22\x68\x74\x74\x70\x3a\x2f\x2f\x6e\x73\x2e\x61\x64\x6f\ +\x62\x65\x2e\x63\x6f\x6d\x2f\x78\x61\x70\x2f\x31\x2e\x30\x2f\x22\ +\x0a\x20\x20\x20\x20\x78\x6d\x6c\x6e\x73\x3a\x64\x63\x3d\x22\x68\ +\x74\x74\x70\x3a\x2f\x2f\x70\x75\x72\x6c\x2e\x6f\x72\x67\x2f\x64\ +\x63\x2f\x65\x6c\x65\x6d\x65\x6e\x74\x73\x2f\x31\x2e\x31\x2f\x22\ +\x0a\x20\x20\x20\x78\x61\x70\x52\x69\x67\x68\x74\x73\x3a\x57\x65\ +\x62\x53\x74\x61\x74\x65\x6d\x65\x6e\x74\x3d\x22\x68\x74\x74\x70\ +\x3a\x2f\x2f\x62\x6c\x6f\x67\x2e\x61\x64\x64\x69\x63\x74\x65\x64\ +\x74\x6f\x63\x6f\x66\x66\x65\x65\x2e\x64\x65\x22\x0a\x20\x20\x20\ +\x78\x61\x70\x3a\x4d\x65\x74\x61\x64\x61\x74\x61\x44\x61\x74\x65\ +\x3d\x22\x32\x30\x30\x39\x2d\x30\x31\x2d\x32\x32\x54\x31\x30\x3a\ +\x30\x36\x3a\x31\x34\x2b\x30\x31\x3a\x30\x30\x22\x3e\x0a\x20\x20\ +\x20\x3c\x64\x63\x3a\x63\x72\x65\x61\x74\x6f\x72\x3e\x0a\x20\x20\ +\x20\x20\x3c\x72\x64\x66\x3a\x53\x65\x71\x3e\x0a\x20\x20\x20\x20\ +\x20\x3c\x72\x64\x66\x3a\x6c\x69\x3e\x4f\x6c\x69\x76\x65\x72\x20\ +\x54\x77\x61\x72\x64\x6f\x77\x73\x6b\x69\x3c\x2f\x72\x64\x66\x3a\ +\x6c\x69\x3e\x0a\x20\x20\x20\x20\x3c\x2f\x72\x64\x66\x3a\x53\x65\ +\x71\x3e\x0a\x20\x20\x20\x3c\x2f\x64\x63\x3a\x63\x72\x65\x61\x74\ +\x6f\x72\x3e\x0a\x20\x20\x20\x3c\x64\x63\x3a\x64\x65\x73\x63\x72\ +\x69\x70\x74\x69\x6f\x6e\x3e\x0a\x20\x20\x20\x20\x3c\x72\x64\x66\ +\x3a\x41\x6c\x74\x3e\x0a\x20\x20\x20\x20\x20\x3c\x72\x64\x66\x3a\ +\x6c\x69\x20\x78\x6d\x6c\x3a\x6c\x61\x6e\x67\x3d\x22\x78\x2d\x64\ +\x65\x66\x61\x75\x6c\x74\x22\x3e\xef\xa3\xbf\x20\x4d\x61\x64\x65\ +\x20\x6f\x6e\x20\x61\x20\x4d\x61\x63\x21\x26\x23\x78\x41\x3b\x26\ +\x23\x78\x41\x3b\x65\x78\x63\x6c\x75\x73\x69\x76\x65\x20\x66\x6f\ +\x72\x20\x53\x6d\x61\x73\x68\x69\x6e\x67\x20\x4d\x61\x67\x61\x7a\ +\x69\x6e\x65\x3c\x2f\x72\x64\x66\x3a\x6c\x69\x3e\x0a\x20\x20\x20\ +\x20\x3c\x2f\x72\x64\x66\x3a\x41\x6c\x74\x3e\x0a\x20\x20\x20\x3c\ +\x2f\x64\x63\x3a\x64\x65\x73\x63\x72\x69\x70\x74\x69\x6f\x6e\x3e\ +\x0a\x20\x20\x20\x3c\x64\x63\x3a\x73\x75\x62\x6a\x65\x63\x74\x3e\ +\x0a\x20\x20\x20\x20\x3c\x72\x64\x66\x3a\x42\x61\x67\x3e\x0a\x20\ +\x20\x20\x20\x20\x3c\x72\x64\x66\x3a\x6c\x69\x3e\x69\x63\x6f\x6e\ +\x73\x3c\x2f\x72\x64\x66\x3a\x6c\x69\x3e\x0a\x20\x20\x20\x20\x20\ +\x3c\x72\x64\x66\x3a\x6c\x69\x3e\x66\x6c\x61\x76\x6f\x75\x72\x3c\ +\x2f\x72\x64\x66\x3a\x6c\x69\x3e\x0a\x20\x20\x20\x20\x20\x3c\x72\ +\x64\x66\x3a\x6c\x69\x3e\x73\x6d\x61\x73\x68\x69\x6e\x67\x20\x6d\ +\x61\x67\x61\x7a\x69\x6e\x65\x3c\x2f\x72\x64\x66\x3a\x6c\x69\x3e\ +\x0a\x20\x20\x20\x20\x20\x3c\x72\x64\x66\x3a\x6c\x69\x3e\x61\x64\ +\x64\x69\x63\x74\x65\x64\x20\x74\x6f\x20\x63\x6f\x66\x66\x65\x65\ +\x3c\x2f\x72\x64\x66\x3a\x6c\x69\x3e\x0a\x20\x20\x20\x20\x3c\x2f\ +\x72\x64\x66\x3a\x42\x61\x67\x3e\x0a\x20\x20\x20\x3c\x2f\x64\x63\ +\x3a\x73\x75\x62\x6a\x65\x63\x74\x3e\x0a\x20\x20\x20\x3c\x64\x63\ +\x3a\x72\x69\x67\x68\x74\x73\x3e\x0a\x20\x20\x20\x20\x3c\x72\x64\ +\x66\x3a\x41\x6c\x74\x3e\x0a\x20\x20\x20\x20\x20\x3c\x72\x64\x66\ +\x3a\x6c\x69\x20\x78\x6d\x6c\x3a\x6c\x61\x6e\x67\x3d\x22\x78\x2d\ +\x64\x65\x66\x61\x75\x6c\x74\x22\x3e\xc2\xa9\x20\x32\x30\x30\x39\ +\x20\x62\x79\x20\x4f\x6c\x69\x76\x65\x72\x20\x54\x77\x61\x72\x64\ +\x6f\x77\x73\x6b\x69\x20\x3c\x2f\x72\x64\x66\x3a\x6c\x69\x3e\x0a\ +\x20\x20\x20\x20\x3c\x2f\x72\x64\x66\x3a\x41\x6c\x74\x3e\x0a\x20\ +\x20\x20\x3c\x2f\x64\x63\x3a\x72\x69\x67\x68\x74\x73\x3e\x0a\x20\ +\x20\x3c\x2f\x72\x64\x66\x3a\x44\x65\x73\x63\x72\x69\x70\x74\x69\ +\x6f\x6e\x3e\x0a\x20\x3c\x2f\x72\x64\x66\x3a\x52\x44\x46\x3e\x0a\ +\x3c\x2f\x78\x3a\x78\x6d\x70\x6d\x65\x74\x61\x3e\x0a\x3c\x3f\x78\ +\x70\x61\x63\x6b\x65\x74\x20\x65\x6e\x64\x3d\x22\x72\x22\x3f\x3e\ +\xee\x04\x51\xd2\x00\x00\x00\x19\x74\x45\x58\x74\x53\x6f\x66\x74\ +\x77\x61\x72\x65\x00\x41\x64\x6f\x62\x65\x20\x49\x6d\x61\x67\x65\ +\x52\x65\x61\x64\x79\x71\xc9\x65\x3c\x00\x00\x0d\x9f\x49\x44\x41\ +\x54\x78\xda\x62\xfc\xff\xff\x3f\x03\x36\xc0\x02\x63\xdc\xf5\x72\ +\xff\xcf\xc4\xc2\xc2\xf0\xef\xcf\x1f\x06\xe5\x6d\x3b\x19\x19\x41\ +\x3a\x40\x82\xec\x82\x02\x0c\xcc\x1c\x1c\x0c\x7f\x7f\xfc\x60\xf8\ +\xf9\xfe\x03\x03\x13\x48\xf5\xaf\xd7\xaf\x19\xbe\xbc\x7f\xc7\x10\ +\x7e\xec\x24\xc3\x5f\x20\x0d\xe2\x83\x25\x40\xda\x59\xbf\xff\x60\ +\x58\xac\x28\xcb\xf0\x07\x48\x83\xf8\x8c\x30\xcb\xaf\x1a\xea\xc1\ +\xed\xd0\x3e\x7f\x89\x11\x20\x80\x18\xf1\xba\xea\xb6\x9b\xd3\x7f\ +\x66\x76\x0e\xb0\xc0\xee\xaf\x3f\x3f\xa5\xef\xdb\xcb\xcf\x14\x96\ +\x98\x39\x9f\x8d\x97\x87\x81\x43\x44\x18\x8c\x3d\x04\xb9\xf9\x40\ +\x0a\x18\x4f\x68\x6b\xfc\x11\x16\x16\x62\x4e\x7a\xfe\x0a\xac\x63\ +\x9e\xa4\x18\x83\xee\x95\x07\x07\x19\x39\x84\xa4\x94\x2e\xa8\x49\ +\xdd\x65\xe7\x03\x2b\x64\xf8\xf9\xe9\x13\x83\xfa\xf1\xd3\x8c\x4c\ +\xdf\xdf\x3e\xbd\x37\xf9\xf9\xdb\xc7\xdf\xde\xbe\x63\x00\x61\x83\ +\x5b\xcf\xe6\x83\x14\x00\x04\x10\x86\xab\x40\x76\x06\x3f\xbc\x19\ +\x6a\xc4\xc2\xc0\xbd\xf5\x17\xe3\x87\x97\x0e\x4e\x6e\xed\x0d\xd5\ +\xa7\x61\xf2\x70\x0d\x60\x17\xb3\xb1\x33\x30\xb3\xb3\x32\x30\xb2\ +\x00\x31\x13\x13\xc3\xff\x7f\xff\x18\xfe\xff\xf9\xcd\xf0\xf7\xe7\ +\x6f\x86\xd5\x5f\x7e\xbd\x2f\x3f\xb0\x57\x08\xac\x61\xb5\xa3\xf3\ +\x57\x63\x1e\x76\x2e\x16\x2e\x4e\x86\xbf\x5c\x5c\x0c\x15\xb7\xee\ +\x32\x30\x32\x03\x35\xfc\xfd\xc7\xd0\xaf\xa6\xcc\xf0\xe7\xdb\x37\ +\x20\xfe\xce\x90\xc3\x25\xb6\x09\x1c\x1e\x27\x1e\x3c\x78\xaf\x2d\ +\xcc\xc7\xc5\xca\xc9\xc1\xf0\x07\xa8\xe1\xd9\xcb\x97\x70\x1b\x7e\ +\xf1\x70\x32\xfc\x06\x6a\xf8\x0d\x0c\xf2\xfd\xf7\x2f\xae\x87\x3b\ +\xa9\x56\x49\xe5\x59\xa4\x20\x8f\x24\x33\x1b\x2b\x03\x28\xe8\x61\ +\x1a\x40\x51\xf0\xf7\xd7\x6f\x06\xe3\x07\xaf\xe7\x03\x03\x22\x89\ +\x01\xa4\x01\x86\x81\x00\x14\xec\x7a\x6c\xfc\xa2\x4b\xd9\x05\x25\ +\x6e\xb3\xf1\x0a\x75\x03\xf9\x6a\xc8\x6a\x00\x02\xd0\x4d\xfe\x2e\ +\x0d\x03\x51\x1c\x7f\xc9\xe5\x87\x41\xd4\x4a\x45\x05\x15\x51\x47\ +\x3b\x09\x2d\x82\x48\x29\x1d\x5c\xfc\x03\x9c\x5c\x6a\x27\x27\x9d\ +\x9d\xdc\x1c\xdc\x05\x37\xc7\xee\x6e\x12\x8b\x6b\x0a\x0e\xc5\xa9\ +\x6e\x22\xed\x24\x45\xa8\x92\xdc\x2f\xdf\xdd\x95\xa3\x06\x1a\x38\ +\x92\x7b\x3f\xf2\xde\xbd\xcf\xf7\xa6\xb2\x9b\xf6\x78\x79\x43\xf3\ +\xb0\xde\x3f\x0b\xe5\xd2\x2f\xaa\xe6\x82\x45\xef\xaf\xed\xc7\xdd\ +\x49\xbf\xad\xb0\x57\x3b\xee\xb6\xc2\xb4\xe4\x10\x02\xae\xab\x65\ +\x06\x42\x8d\x95\x73\xb8\xaf\x54\x6b\x37\xd7\x57\x6d\x5b\xe1\xa4\ +\x71\x7e\xd7\x9a\xc9\x4a\x24\x08\xc0\x45\x16\x2e\x26\xe9\x04\x0c\ +\x16\x59\x0a\xcd\xe4\xe5\x59\xfd\xdc\x56\xe8\x1d\xd5\xa5\xa7\xa0\ +\x21\x07\x12\x62\x02\x82\x33\x4a\x56\xd0\x52\xe0\xc8\x20\x1e\x65\ +\xdf\x8d\xf8\x69\x41\x57\x70\x5c\x02\x8e\xef\x81\xeb\xfb\x70\xd9\ +\x53\xd0\xcc\xd1\x24\x67\x70\xbb\xb3\x05\xc2\xa7\x50\x0d\xd9\xbc\ +\x6d\x29\x1d\x0c\x80\x23\x34\xa9\xa0\xe1\x45\x71\xc6\x15\x94\x2c\ +\xe8\x6c\x04\x99\x22\x8d\xe0\x6c\x82\x9e\xb1\x90\x1a\x14\xe0\x1b\ +\xb0\x77\xd3\x93\xb1\x69\xdf\x78\x38\x3a\x81\x0b\x21\x09\x63\x0e\ +\xa7\x14\x1e\xb6\x37\x75\x6b\x3a\x1e\xf7\x0c\xcf\x20\x91\x76\xf9\ +\xe3\x2b\xc1\x51\x9b\x6b\xba\xff\x39\x0c\x39\x1a\x95\x53\x69\x86\ +\x8e\x7e\xcc\xc2\x6f\x65\x53\x3e\x94\x45\xe5\x1f\x87\xa8\xb8\xb6\ +\x9a\xac\x2f\xf6\xf3\x1c\x86\x94\xd1\x83\xee\x5b\x60\xc9\xe5\xb4\ +\xb4\xe1\x45\x73\xa7\x61\x61\xa5\x13\x14\x96\x63\xdc\x97\x71\x15\ +\x27\x63\xfe\x04\xa0\xad\x6a\x42\x21\x08\xc3\xf0\xcc\xce\x36\xab\ +\x6d\x57\x14\x91\xa3\x28\x07\x07\x07\x6d\x8a\xa2\xdd\x83\x9f\x08\ +\x57\x29\xb9\x70\x91\xe4\xa0\xad\x55\x28\xe5\xe2\xe0\xe8\xe0\xe2\ +\x28\xe5\x82\xb8\x39\xe1\xe8\xa4\x36\x1c\x14\xed\x16\x89\x35\xbb\ +\x96\x9d\xf9\x66\x3d\xef\xfb\xed\x2c\xfb\x43\x39\xd8\x7a\xf7\xdd\ +\x6f\x67\xbe\xe7\x99\xef\x7d\x9f\xe7\x9d\x3f\x6b\xe9\xaf\x1f\xf7\ +\x4f\x17\xc6\xa7\x66\x37\x02\x57\x97\x13\xfd\x1e\xa5\x4a\x55\x71\ +\x24\x55\x65\x71\x50\x92\xcf\x94\xe5\x1f\x36\x4a\x3e\x9a\xd4\x6e\ +\xc6\x86\xfb\xda\x22\x0b\xf3\x6f\xc5\x38\x25\x27\x20\xcd\xee\xe8\ +\xe9\x56\x16\x0b\x6a\x45\xf5\x52\xdd\x44\x80\x4c\xe8\xaa\x83\x4d\ +\x5f\x34\x22\x6c\xa9\x69\xc4\xe9\xbb\x65\x44\xbb\x83\x81\xb5\xe5\ +\xc5\x68\x59\x82\xad\x60\x28\xd1\xa3\xbb\x2a\xb9\x11\x6e\x4d\x1a\ +\x9b\x42\x93\x64\x0a\x91\x49\x7c\xea\x90\xd4\x0c\x34\x44\x32\xb0\ +\x39\x04\xaf\x67\xfc\x0d\x07\xc7\xbb\xdb\x43\x05\x04\x93\x9d\xa1\ +\x78\xc4\xef\xaa\x27\x30\x4d\x27\x60\x18\x09\x72\x37\x41\x10\x8e\ +\x5e\xf3\x8c\x52\xb8\x54\x39\x06\x7a\x7a\x9a\x59\x2d\xcd\x12\xdc\ +\xb4\x40\x94\xc1\x74\xb1\x98\x64\xb3\xbd\x6b\x70\x7d\x75\xe9\xd0\ +\xe5\x3c\xfd\xb4\x30\x6a\xac\x54\x4a\xb1\x21\x7b\x91\x4a\x73\xb6\ +\xd3\xc8\x78\xc1\x24\x39\x3e\x94\xe4\x07\x45\x46\x66\x5a\xe3\xff\ +\x2c\x82\xef\xfb\xb6\x8f\x70\xf6\xf6\x8f\x22\x05\x4d\x7e\x4c\xbc\ +\x66\x7c\x1e\xdd\x9d\xa5\xd2\xc0\x09\x1a\x22\x8b\xf9\x67\x21\x1b\ +\xd8\xe0\x94\x28\xdf\xe5\x5c\x89\x84\x61\x28\xe4\x20\x1b\x73\x91\ +\x33\x42\xa0\x54\xf1\xfb\xa7\x5b\xc5\x71\x0e\xbf\x10\x1e\x8c\x33\ +\x3a\xb6\x6c\x98\x90\xb3\x85\x8f\x6e\xca\x3a\xe7\x6a\x4d\x06\xe7\ +\x9c\xfb\x8f\xae\xf3\x7d\xb4\xa6\x7d\xd8\x9f\x10\xc2\x24\x21\x96\ +\x34\x19\x6e\x3b\x87\xdb\x3a\xf2\x4d\xc6\x98\xa1\x9c\x77\x5f\x51\ +\x93\x9d\x89\x48\xcd\xb5\x85\x6c\x72\xd2\xb4\xac\x9e\xd8\x4b\x23\ +\xac\x7c\x57\x56\xa6\x20\x69\x5a\xa9\xf6\x9e\x0d\xf8\x2a\x6a\xf3\ +\x32\xd5\x7e\x91\xa9\xf8\x92\x29\x06\xca\x09\x80\x83\xbf\xfa\xa0\ +\x88\x2c\x1c\xa9\xf2\xce\x8d\xf8\x3c\x75\xe5\x8c\xf6\x2a\x6c\x2b\ +\x14\x7b\xb9\xc0\xa2\x17\xc0\xcf\xe5\x30\xfe\x7d\x54\x7c\x0a\x40\ +\x7c\xb5\x85\x44\x15\x84\xe1\x39\x97\x3d\x95\xe6\xda\x16\x3d\x58\ +\xf6\x12\x62\x17\x8a\x88\xb0\x20\xc2\x32\x08\xa4\xde\xba\x18\x41\ +\x20\xbd\xd4\x53\x2f\x11\x84\x10\x62\xa0\x20\x3d\x44\x10\x54\x04\ +\x5d\xe8\x46\xf8\x52\x52\xc4\x1a\xe5\x16\x84\x94\x41\xf8\x50\xb1\ +\x5a\x64\xa0\x68\x65\xa4\xdb\x6e\x7a\xf6\x5c\xe6\xf4\xff\x73\x39\ +\xbb\xb3\x9a\xe9\x83\xb4\x30\x7b\xfe\x73\x66\xe6\xff\x0f\x33\xdf\ +\x7c\xdf\x77\xe6\xbc\x80\x4e\xe6\xf8\x67\x4e\xd7\xd9\xd0\xd4\x52\ +\x15\x7f\xde\x75\xa3\x3e\xb0\x97\x6d\x88\x04\x45\xe5\x3a\xb1\x7e\ +\x52\xe2\x7d\xa1\x24\xfb\xd4\xd5\xd2\xa9\x8a\xca\x78\xdb\xf5\x4b\ +\x47\x66\xbd\x44\x2d\x67\xcf\x15\xdd\x6d\x8f\xf7\xdc\x2f\xf6\x2a\ +\x34\xf6\x33\x18\x7a\x18\x8a\x88\x44\x10\x5e\x7d\x12\xcf\x92\xb1\ +\x57\x95\x6b\x6f\xdf\xb9\x72\xe1\xf8\x8c\x0a\x9c\x6a\x6c\xde\xb1\ +\xf2\x65\xa2\x7d\x67\x84\x44\x25\xa3\x72\x78\x72\x04\xf1\x02\xfc\ +\x8f\x8b\x27\x3f\x98\x07\x9d\x05\xef\xde\x26\x1e\xad\x9f\xb6\x40\ +\x43\x53\xf3\xea\x55\x2f\x3a\xbb\xb7\x59\x46\x09\x62\x9f\x25\x17\ +\xe7\x80\x51\x85\x26\x65\x8d\x08\x26\x15\x8c\x2a\xe2\x3a\x67\xde\ +\xa4\x22\xca\x26\x2f\x4a\x24\xba\xc2\xe4\x78\x7a\xd1\x3c\x58\x68\ +\x20\x2c\x30\x12\x11\x76\x55\x62\xd6\x27\xe8\x1c\xe6\xb4\x59\xd9\ +\x75\x68\x6b\xa6\x2c\x50\x7b\xa0\xfe\xe1\xde\xf9\x24\x16\x26\x47\ +\x8a\x30\x91\x2a\x38\x6d\xeb\x2c\x69\x7e\xe3\x7d\x7c\x8c\x11\x16\ +\xd9\xd5\xdf\x7b\x68\xca\x02\xfb\x7f\x0c\x56\xa3\x34\x86\xcb\xc2\ +\x26\x9a\xcc\x02\x69\x82\x5d\x65\x72\x8c\x35\xd6\xcc\x9c\x28\x89\ +\x79\xdb\x2d\x2d\x8a\x4e\x5a\x81\xe9\xc9\xd3\x67\xf6\x1c\x63\x9b\ +\xca\x37\x94\xbd\x91\xce\xdf\xea\x44\xb2\x8f\xc5\xc4\x30\x94\x3d\ +\x40\x37\xd4\xba\xa6\x92\x44\xd0\x39\x80\x13\xd2\x4d\xce\x49\x98\ +\x63\xe1\xa7\xbe\x5a\xa5\xc0\x87\x8f\xfd\x47\xd9\x64\x84\x89\x64\ +\x4d\xb1\x54\x44\x26\x67\x6f\x28\x60\x4a\x39\x30\xd8\xb2\xb0\x02\ +\xb8\xd1\x42\x2f\x20\xde\x6d\x91\xa8\x52\x20\x99\x4c\x2e\xf5\x8b\ +\xa1\x2f\x62\x70\xa1\x71\xa0\xa1\xfd\x82\x96\x01\x85\x22\x0c\x45\ +\x58\x44\xcb\xf3\x70\x40\xd1\xa8\x80\x6e\x4e\x68\x64\x5c\xee\xf8\ +\x96\x52\x60\x78\x70\xc0\xf1\xca\x63\x6c\x3d\x03\x68\x52\xcd\x88\ +\x50\x33\x12\x8a\xbe\xa6\x98\x44\x2f\x9d\x21\xba\xe7\xaa\x8a\x06\ +\xe2\xa3\xbb\xae\x56\x48\x15\x43\x21\xc7\x43\x10\x08\xcf\x43\xe4\ +\x19\x91\x71\x40\xd4\xfb\xbc\x71\x01\x11\x73\x21\xfe\xee\xd1\x6c\ +\x21\x8a\x9e\x04\x62\xb6\x72\x4a\xa5\x45\x94\x57\xca\x9b\x7c\x46\ +\x51\x70\x94\x31\x3c\x59\x62\xc2\x19\x2d\x2c\x70\x33\xed\x07\x5e\ +\x4e\xcc\x03\x7e\x4a\xd1\x37\x33\x0f\x4d\xc5\x73\xae\xbb\xe1\xbd\ +\xf0\x41\xfc\x44\xf3\xb9\x98\xe3\xe2\x2f\xfb\xfd\x24\xaa\x00\xf5\ +\xea\x06\x3d\xae\xd2\x85\xd1\x32\x04\xc6\xe5\xbd\x5e\xb0\x07\x39\ +\x3d\xe6\xa6\xcb\x77\xb9\x11\xf0\x3c\x2f\xd8\x32\x38\x5a\x81\x9f\ +\x9f\x85\x74\xbd\x2f\x43\xe9\xe7\x12\x4a\x4d\x84\x2b\x0a\xb8\xc4\ +\x3c\x83\xa1\x24\xbd\xf0\x0b\x41\xe8\xb1\x10\x7b\xc9\x49\x90\xfc\ +\xb5\x4c\xae\x9c\x64\x74\x01\x35\x43\xa9\x5b\x72\x69\xa8\x70\x0b\ +\xbe\xe7\x32\xd7\xe6\x0b\x08\xfa\x21\x2c\x3d\xde\xe7\xf1\xb1\x38\ +\xe7\x5e\x6a\x62\x18\x52\x55\x4f\x4b\xd7\xb0\x54\x9d\x6f\x96\xc7\ +\x6a\x14\x36\xd5\xff\xc2\xa6\x79\xed\x71\xda\x1e\x69\x1c\xfd\xbd\ +\x15\x5e\xf4\xd3\x3f\x05\x07\x8a\x5c\xeb\x28\x2b\x3d\xbc\xd8\x34\ +\x22\x33\xd1\x83\xcb\x63\xe3\x03\x57\xd3\xf6\x66\x48\xfe\x75\xc6\ +\xa2\x0f\x45\x62\x70\xe9\x78\x56\x56\xba\x31\x6a\x80\x03\x9b\x42\ +\xd1\x1e\x64\xec\x6f\x2d\x63\xe3\xe7\x21\x71\xeb\xac\x5d\x05\x24\ +\x5b\x82\x9f\x43\xd0\x4a\xe1\x33\xbb\x8e\x18\xe6\x26\x60\xdb\x15\ +\x90\x3d\x15\x04\xb4\x97\x3a\x76\x87\x37\x91\xee\x81\x7e\xc4\xfc\ +\x08\xe4\xb1\xff\x8b\x6d\xf9\x23\x40\xb7\xd6\x1a\x1b\x45\x11\xc7\ +\xe7\x71\xd7\xbb\x6b\xbb\x57\xda\x42\xb5\x68\x63\x23\x31\x7c\x80\ +\xd8\x68\x50\x1b\x84\x44\x94\x48\xa8\x21\x84\x44\x4c\xfd\x00\xbe\ +\x3e\xf8\xfa\x60\x42\x02\x34\x94\x68\x31\x80\x05\xa3\x89\x1f\x54\ +\x62\x34\xf1\x89\x8d\x69\x02\x86\x40\x03\x51\x31\x01\x0d\x12\x40\ +\x4b\xa8\xda\x40\x4b\x51\x22\x5a\xa5\xf6\xee\xf6\xda\x7b\xec\xce\ +\x38\x33\x3b\x3b\x3b\xbb\x77\x57\x69\x2b\x90\xd8\x74\x6e\x6f\x6f\ +\xef\xf6\xfe\xbf\x99\xff\xeb\xf7\x9b\xbb\xea\x5f\x70\x5d\xdb\xa2\ +\xc9\xfc\xad\xdb\xd4\xb1\x72\xf8\xaf\x91\x25\x7f\x27\x53\x0d\xa3\ +\x89\x54\x7d\x2e\x9f\x8b\x26\x53\x66\x0d\xbf\x16\x37\x2a\x47\xca\ +\xc2\x65\x99\x19\x55\xc6\xa5\xea\xb8\xf1\x6b\xdd\xcc\x9a\xc3\xaf\ +\x6f\xef\xf8\xfc\xba\xad\xc0\x8b\x5b\x3b\x1b\xcf\xf4\x0f\x6e\xeb\ +\x3b\x3b\xb0\x70\x7d\x0c\xd4\xdc\x89\x69\x85\x81\x58\x6f\x06\xb5\ +\x1b\x03\x2d\x2c\xdd\xd0\x04\x44\x9d\x9a\x2c\x45\x9c\x24\x20\xfd\ +\xea\x18\x18\x99\x77\xdb\x9c\x6f\xe7\xcf\xbd\xb5\xfd\xe5\xcd\x6d\ +\x43\x57\x0d\xc0\xa6\x8e\x6d\xf3\x8f\x9e\xe8\x7d\xfb\xa1\xd4\xe5\ +\x79\x4b\xc3\x34\x1e\xc7\x10\x3b\xc1\x82\x64\x88\x43\x37\x7a\xf4\ +\x40\xf2\xe9\x0f\xda\x89\x16\xad\x3c\x1d\x10\x60\x52\x60\x1f\xca\ +\x81\xe4\xfe\xca\xda\xbe\x45\x0b\x9a\x9e\xdd\xde\xd1\x7e\xe6\x3f\ +\x01\xc0\xfb\xd4\x9e\xaf\x8f\xee\x5a\x87\x73\xb3\xef\x0d\x43\x43\ +\x19\x8c\xa0\x67\xb8\xf8\x77\xce\x5d\x00\x10\x01\x10\x5c\x02\x4a\ +\xfc\xe5\xd6\x29\xcb\x1a\x10\xe2\x1c\xbf\xc9\xd9\xa9\xd7\xec\xb2\ +\xdf\x96\xdf\xb7\xe8\x19\x57\x09\x9b\x34\x00\xde\xbd\x1f\x3a\x72\ +\xac\xa7\x35\x75\xb9\x69\x69\x19\xa8\x12\x86\x23\xc7\x60\x88\xb4\ +\x59\xe7\x47\xac\x71\x43\x61\xb9\x0b\xca\x05\x21\x8b\xbd\xd2\x4f\ +\x88\x32\x9c\x97\x1e\xa0\x97\x74\x42\x54\xe9\xff\x22\x4f\x12\x9f\ +\x56\xd4\xf6\x3e\xb8\xb8\x79\x79\x31\x8d\xa5\x24\x00\xce\x69\xf6\ +\x1c\xfc\xaa\xeb\x33\x83\x36\xc6\x98\xe5\x9c\x7a\xf0\x19\x57\x86\ +\xcb\xe7\x4a\x1a\x11\xc6\x23\xff\xaa\x04\xdc\xa9\xc0\x6d\x88\x24\ +\xc0\x9a\x5e\xe0\x4a\x71\x0a\x08\x7b\x3e\xc6\x10\x3e\x92\x04\x43\ +\xab\x96\xdd\xdf\xaa\x6b\xc4\x25\x01\xb4\xbd\xb4\x75\xf1\x9e\x83\ +\x87\x3f\xdc\x17\xa7\xb7\x08\xda\xa4\xd1\x1b\x1f\x0b\x29\xa8\xa3\ +\x50\x03\x03\xa5\xa0\xe9\xad\x00\x70\x1b\x36\xd5\x5c\x11\xcd\x50\ +\x22\xb5\x45\xe2\x67\x34\xf2\x3a\x21\x36\x5d\x91\x00\x17\x56\x2d\ +\x5b\xb2\xb6\x73\xcb\xe6\x23\x25\x01\x30\xb7\x09\x75\x1f\xf8\xb2\ +\xf7\xbd\xd0\xf8\x5c\x83\x05\xa9\x32\x5e\xb8\x0e\xe7\x0a\xee\x4a\ +\xb8\x6d\x3d\x02\x19\xf6\xf9\x8d\xc7\x8e\x03\xcd\xf1\x81\x93\x80\ +\x60\xc1\xca\x0a\x47\x52\x71\x40\xc0\xce\x85\xcd\x20\xc2\x01\x2b\ +\xa3\x6d\x39\xa8\xd3\xcc\x51\xa2\x84\xd3\x94\x6d\xdb\x4f\xe6\xa2\ +\xfd\x0f\xb7\x3c\xd0\xc4\xdc\xc9\x2a\x5a\x07\x7e\x1e\xfc\x65\xe7\ +\x5a\x90\x99\x2d\x8c\xe7\x6e\x23\xdd\x41\x18\x8f\x25\x10\x14\xf2\ +\x56\x41\x36\x7a\x78\xe6\x2c\x67\xe6\x11\x94\x62\x58\x69\x17\x02\ +\x52\x3b\xe1\x3e\xce\x55\x69\xd1\xcb\x15\xac\x98\x25\x98\x02\xe0\ +\x2d\x1f\x24\xe2\x3e\x06\xc2\x78\x0d\xc8\xd4\x7f\x3f\x78\xe1\x15\ +\xf6\xea\xfa\xa2\x00\x7e\xea\x3f\x7b\xd7\x13\x24\x17\xa3\xbc\xa9\ +\x47\x44\xb8\x04\x95\x2e\x82\x28\x76\x7c\x00\xe5\xd9\x51\xf2\x1c\ +\x6e\x0c\x3b\xe6\x59\xd3\x28\xf8\x0f\xe5\x00\xa8\xe2\x3f\x22\x2e\ +\x54\xec\x12\xb1\x2a\x44\xba\x8e\x70\x1f\x67\x3f\xc7\x69\x9b\x5d\ +\x59\x51\xb5\xd0\x44\x4a\x5b\x9e\x6b\xdd\x41\xad\xf2\xdd\xfd\xe7\ +\x9a\x4b\x56\xe2\xf3\x03\xe7\xb0\x51\x1b\x09\x59\x18\x6b\xfe\x0d\ +\x85\x71\xc8\xed\x1f\xb1\x27\x7b\xf1\x95\xe0\xef\x4d\x98\x69\x35\ +\xfb\x6e\x0c\xa8\xd9\x84\x7a\xf6\xd1\x62\x80\x01\xb2\x92\x29\x80\ +\xa9\x67\x34\xd1\x41\xf8\xdc\xca\x71\x23\xc3\xb6\x43\xe7\xff\x1c\ +\xc6\x25\x01\x8c\xa5\xcd\xd1\x71\x03\x91\x58\x88\x60\x1d\x00\x44\ +\xfc\xe6\x8e\xa4\xa6\x08\x87\x78\x8e\xc4\x97\xe6\x59\x87\x3d\x15\ +\x00\x94\x8b\x8c\xd4\x3f\xeb\x4a\x60\x2c\x02\x60\xcc\xb2\x09\xb7\ +\x71\x22\x0d\xea\x87\xbe\x9c\x6d\x52\x1f\x2d\xd3\x15\x1b\xe0\xa3\ +\x69\xc2\xad\x89\xc6\x09\xb5\x7c\xae\x5e\x0f\x1c\xa9\xc6\x21\xa9\ +\x8c\x85\xa2\xf7\x75\xeb\x04\xf5\x76\x11\x7e\xcc\xe6\x4d\x6e\xe3\ +\x44\x00\xba\xba\xd2\xd9\x3f\x94\x40\x40\x35\x62\x2a\xd2\x1a\xd1\ +\xb6\x4e\xa4\x9f\x12\x2f\x97\x03\x3d\x15\x96\x18\xc1\xf7\x91\x82\ +\xeb\x54\x66\x1f\xe2\xfb\x7e\x6e\xcf\xc7\xa6\xb0\xed\x93\x92\x2e\ +\xc4\x98\xc3\x69\xc6\x34\xde\xdd\x97\xce\x6e\x58\x51\x11\xa9\x63\ +\x7e\xa2\x65\x0d\x27\xab\x10\x96\x1a\x10\xf0\x58\x78\x8c\xb9\xd2\ +\xde\xd6\xd5\x22\x98\x39\x05\x05\xbe\x02\x87\xb4\x24\xe4\x19\x0d\ +\xe4\xc6\x9f\x79\xea\x94\x0a\x54\xd7\x5d\x08\xf1\x52\x27\xd5\x88\ +\x37\x67\x37\x27\xb3\xd6\x3b\xcc\xc6\xbe\x2b\xe1\x64\x3b\xda\x67\ +\xc4\x1e\x5b\x59\x11\xb9\x41\xc5\x02\x84\x01\x12\x18\x38\x9f\x42\ +\x21\xa3\x81\x42\x46\x03\xab\xeb\x5e\xdf\x9b\x12\xd4\xec\x03\x66\ +\xfc\xc6\xc9\x70\xbe\xe7\x9e\x32\x22\x6d\x4f\xc7\x63\x0d\x10\xa2\ +\xc2\x56\x02\x6a\x40\xa6\xd9\x4a\xf8\x66\x3c\xd0\x4a\xec\x4a\x08\ +\xc2\xda\xc9\x8c\x7f\x6b\xd2\xdd\x28\x03\x51\xc7\x0e\xef\x6f\xa9\ +\x2e\x5f\xd0\x52\x1e\x99\x55\xb2\x99\x43\x5e\x33\x07\xa5\x26\x3f\ +\x51\x33\xe7\x25\x05\xaa\x7c\x3d\xd8\xcc\xf5\x98\x59\x4e\xe3\x4f\ +\xb0\x0f\x3c\xce\x8c\x1f\x9e\x16\x1f\xe0\x3b\x14\xec\xf0\xe6\x86\ +\xaa\xd8\xed\xab\x2b\xa3\x37\x5e\x51\x3b\x0d\x8b\xb4\xd3\xf4\xdf\ +\xdb\xe9\x6e\x33\xf3\xfb\x8e\xd1\xb1\xd3\xec\x5d\xcf\x07\x35\x88\ +\x69\x33\x32\x06\x84\x6f\xca\xae\xe1\xb3\xf2\x42\x55\x6c\xce\xa3\ +\x95\x91\x7a\xec\xf4\x1a\x53\x26\x34\x84\x3d\xec\x4e\x65\x2e\xbd\ +\x91\x18\x1f\x60\x2f\x7d\xc4\x57\x9c\x19\x9e\xbf\x26\xa4\x9e\x01\ +\x6a\xe0\xfa\x30\x1b\x2d\x77\x47\x42\x37\xdf\x13\x0d\xc7\x1b\x43\ +\x28\x7a\x13\x46\xd1\x6a\x8c\xc3\x35\x18\x8a\x5d\xe8\x11\x42\xf3\ +\xa3\x16\xc9\x5f\xb4\xed\xcc\x90\x45\x32\xdf\x65\xf2\xc9\xe3\x59\ +\xeb\x22\xbb\x74\x80\x8d\x6e\x77\x73\xea\x9a\x70\x62\x36\xab\xfc\ +\xa7\x0d\x15\x6c\x94\x73\x0c\xf2\xa7\x0e\x11\x36\xca\xe4\x08\xc9\ +\x81\xb4\x3a\x43\xe4\xb0\xe4\xc8\xc9\xc1\x85\x4e\xae\xc7\xf0\x8d\ +\x75\x4e\x58\xd2\xa5\xf4\x99\xff\xad\xac\xf2\x0f\x8b\x79\x94\xf0\ +\x0c\x3c\x66\x86\x00\x00\x00\x00\x49\x45\x4e\x44\xae\x42\x60\x82\ +\ +\x00\x00\x38\x97\ +\x89\ +\x50\x4e\x47\x0d\x0a\x1a\x0a\x00\x00\x00\x0d\x49\x48\x44\x52\x00\ +\x00\x00\x89\x00\x00\x00\x87\x08\x06\x00\x00\x00\x22\x2e\x7a\xb9\ +\x00\x00\x00\x06\x62\x4b\x47\x44\x00\xff\x00\xff\x00\xff\xa0\xbd\ +\xa7\x93\x00\x00\x00\x09\x70\x48\x59\x73\x00\x00\x0b\x13\x00\x00\ +\x0b\x13\x01\x00\x9a\x9c\x18\x00\x00\x00\x07\x74\x49\x4d\x45\x07\ +\xe1\x01\x02\x0c\x02\x23\x1f\xee\x69\x28\x00\x00\x20\x00\x49\x44\ +\x41\x54\x78\xda\xed\x9d\x77\x7c\x5c\xe5\x99\xef\xbf\xef\x34\x95\ +\x51\xef\xcd\x45\xc5\xbd\xf7\x8a\xe9\xbd\x13\x0c\x24\x01\xf6\x93\ +\x25\x40\xda\xee\x4d\xf6\x6e\x02\xbb\xb9\xb9\xc9\xde\x9b\x02\x29\ +\x7b\x13\x20\xd9\x18\x02\x4b\x48\xa1\x18\x88\xe9\x45\x54\x1b\xdb\ +\xb8\xcb\xb6\x64\x5b\xb2\x24\xab\x77\x8d\xca\x14\x4d\x3b\xe7\xb9\ +\x7f\xbc\x23\x8d\x6c\x4c\xb1\x3d\xb2\x8d\xe3\xc7\x9f\xf1\x48\xa3\ +\x99\x73\xe6\xbc\xef\xef\x3c\xbd\xc0\x59\x3a\x4b\x67\xe9\x2c\x9d\ +\x28\xa9\xbf\xc7\x8b\x16\xc3\x28\xa1\xb3\xb3\x8e\x40\x00\x82\x41\ +\x08\x85\x20\x1c\x06\xd3\x04\x11\xfd\x26\x8b\x45\x3f\xec\x76\xfd\ +\x70\x38\x20\x21\xa1\x4f\xe5\xe4\x64\x9c\x05\xc9\x99\x06\x08\x97\ +\x6b\x3b\xbd\xbd\x0b\xe8\xef\x87\xc1\x41\xe8\xee\x86\x8e\x0e\xe8\ +\xea\x02\x97\x4b\xbf\xe6\xf1\xc0\xd0\x90\x06\x8b\x61\x80\x52\x60\ +\xb3\x69\x60\x38\x9d\x90\x94\x04\xa9\xa9\x90\x95\x05\x79\x79\x90\ +\x9b\x0b\x19\x19\xfa\xb5\xd4\x54\xd4\x84\x09\xea\x2c\x48\x3e\x6f\ +\xc0\xa8\xad\x15\x7a\x7a\xa0\xb9\x19\x6a\x6a\xa2\x8f\xe6\x66\x18\ +\x18\xd0\xdc\x63\x34\xd7\x18\x7e\x1e\xfe\x59\x1d\xb1\x2c\x4a\x45\ +\x1f\x56\x2b\x24\x24\x40\x76\x36\x14\x17\xc3\xe4\xc9\x30\x65\x0a\ +\x94\x96\x6a\x00\x65\x67\xa3\x0a\x0b\xd5\x59\x90\x9c\x8e\xc0\xa8\ +\xab\x13\x5a\x5a\xa0\xaa\x0a\xb6\x6f\x87\x5d\xbb\xa0\xa1\x01\x7c\ +\x3e\xcd\x1d\x86\x41\x31\x1a\x10\xc3\x00\xb0\x5a\x35\xe7\xb0\x58\ +\xf4\xef\xc3\xef\x33\x4d\x2d\x86\x0c\xe3\x28\x2b\xa7\xa2\x22\xc9\ +\x66\xd3\x9c\x65\xda\x34\x58\xb8\x10\xe6\xcf\x87\x92\x12\x28\x2a\ +\x42\xe5\xe5\xa9\xb3\x20\x39\xd5\xa2\xa4\xa1\x61\x01\x55\x55\xf0\ +\xfe\xfb\xf0\xc1\x07\x9a\x5b\x04\x02\x7a\x83\x4d\x33\xfa\xe6\xb8\ +\x38\x48\x4e\xd6\x0f\xa7\x13\x12\x13\x35\x47\x88\x8b\x8b\xfe\x6c\ +\xb7\x6b\xc0\x88\x68\x70\x04\x83\x1a\x64\x43\x43\xe0\xf7\xeb\x67\ +\x9f\x4f\x8b\x27\xb7\x5b\x3f\x0f\x03\x68\x34\x68\x92\x93\x61\xe6\ +\x4c\x38\xff\x7c\x58\xb2\x04\x4a\x4b\x51\x53\xa6\xa8\xb3\x20\x39\ +\x99\xe0\x68\x6d\x15\xea\xea\x60\xe3\x46\x78\xe3\x0d\xd8\xb3\x47\ +\x6f\x5a\x38\x1c\xe5\x14\xf1\xf1\x90\x9e\x0e\x99\x99\xfa\xb9\xb0\ +\x50\x8b\x87\x09\x13\xf4\xcf\x11\xd1\x40\x41\xc1\xc5\xca\xe1\x78\ +\xeb\x53\xc1\xd8\xd5\xb5\x80\xae\x2e\x68\x6f\xd7\x40\x3c\x74\x08\ +\x1a\x1b\xb5\x8e\xd3\xdb\xab\x1f\x83\x83\x51\xd0\x58\xad\xfa\x51\ +\x50\x00\x2b\x57\xc2\xa5\x97\xc2\xec\xd9\xa8\x39\x73\xd4\x59\x90\ +\x8c\x25\x38\xda\xda\x84\x03\x07\xe0\xad\xb7\xe0\xa5\x97\xa0\xae\ +\x4e\x73\x8d\xe1\x8d\x89\x8f\x87\x9c\x1c\x0d\x80\xe2\x62\x7d\x37\ +\xcf\x99\x03\x33\x66\xa0\x4a\x4b\x63\x7e\xad\x12\x0c\x5e\xc4\xde\ +\xbd\xe5\xec\xdd\x0b\x15\x15\xb0\x7f\x3f\xb4\xb6\x6a\xc5\xb8\xaf\ +\x2f\xaa\x04\x5b\xad\x9a\xbb\xac\x58\x01\xd7\x5c\x03\x8b\x17\xa3\ +\xe6\xce\x55\x67\x41\x12\xeb\xcd\xd8\xbe\xbd\x9c\x57\x5f\x85\xbf\ +\xfd\x0d\xea\xeb\xa3\x22\x45\x29\xcd\x2d\x8a\x8a\xb4\x02\xb9\x6c\ +\x19\x2c\x5f\x0e\x0b\x16\x94\x2a\xab\xb5\xfe\xa4\x7e\xcf\x96\x16\ +\xa1\xa2\x02\x36\x6c\x80\x9d\x3b\x35\xb7\x69\x6b\xd3\x62\x4a\x44\ +\x83\xc5\xe9\x84\x55\xab\xe0\xc6\x1b\x61\xe9\x52\xd4\xd4\xa9\x7f\ +\x97\x6e\x88\xd8\x2e\xfc\xbe\x7d\x22\x0f\x3c\x20\xb2\x64\x89\x88\ +\xd3\x29\x62\xb1\x68\xb5\xd2\x66\x13\x29\x2c\x14\x59\xb9\x52\xe4\ +\x9e\x7b\x44\xde\x7a\x4b\xc4\xeb\xbd\xef\x74\x52\xa4\xe5\xf1\xc7\ +\x45\xbe\xfc\x65\x91\x39\x73\x44\x52\x52\xa2\xdf\xdd\x6a\x15\xc9\ +\xcc\x14\xb9\xfd\x76\x91\x17\x5f\x14\x69\x6f\x97\xb3\x3b\x7d\x3c\ +\x8b\xdc\xd1\x21\xf2\xf2\xcb\x22\x37\xdd\xa4\x17\x74\xf4\x02\x17\ +\x16\x8a\x5c\x78\xa1\xc8\x7d\xf7\x89\xec\xde\x7d\x5a\x2f\xb0\xf8\ +\xfd\x77\xc9\x2b\xaf\x88\x7c\xed\x6b\x22\xf3\xe7\x6b\xb0\x28\xa5\ +\x1f\x36\x9b\xc8\xe4\xc9\x22\xff\xf1\x1f\x22\xdb\xb7\x9f\x05\xca\ +\x31\x73\x8f\x9f\xfd\x4c\x64\xda\x34\x11\x87\x23\xba\xa8\x59\x59\ +\x22\xab\x56\xe9\xbf\xed\xdf\xff\xb9\x5b\x54\x79\xfd\x75\x91\x3b\ +\xef\x14\x99\x31\x43\x24\x21\x41\x83\x5e\x29\x91\xc4\x44\x91\x6b\ +\xae\x11\x79\xee\x39\x91\xae\x2e\xd7\x59\x04\x7c\xda\x42\xbe\xfb\ +\xae\xc8\x97\xbe\x74\x38\xf7\x48\x48\x10\x99\x35\x4b\xe4\xbb\xdf\ +\x15\xd9\xb9\xf3\x73\x7d\xc7\x89\x61\x94\xc8\xd3\x4f\x8b\x7c\xe1\ +\x0b\x22\x45\x45\x9a\x9b\x0c\x73\xc8\xa9\x53\x45\xee\xbf\x5f\xa4\ +\xba\xfa\x2c\x57\x39\xea\xe2\x75\x75\xb9\xe4\xaf\x7f\x15\x59\xbe\ +\x5c\x83\x62\x98\x7b\x14\x14\xe8\x05\x7d\xf1\x45\x91\x60\xf0\xa2\ +\x33\xe6\x7a\xeb\xeb\x35\x47\x5c\xba\x54\x24\x29\x49\x5f\xab\xc5\ +\x22\x92\x9e\x2e\xf2\x8d\x6f\x88\x6c\xde\x7c\x16\x28\x87\x2d\x58\ +\x53\x93\xc8\xaf\x7e\x25\x32\x65\x4a\xf4\xce\x8a\x8b\xd3\xdc\xe3\ +\x3f\xfe\x43\xe4\xe0\xc1\x33\x76\xc1\xe4\x8d\x37\x44\xbe\xf8\x45\ +\x91\xfc\x7c\xcd\x4d\x94\xd2\xd7\x7e\xed\xb5\x22\xaf\xbd\x76\x16\ +\x28\xa0\xe3\x2c\xf2\xfd\xef\x6b\xd6\x3b\x2c\x5e\x52\x53\x45\x2e\ +\xbe\x58\xcb\xe8\x33\x88\x7b\x7c\xa2\x25\xf4\x83\x1f\x88\x4c\x9f\ +\xae\x75\xb0\x61\xf1\xb3\x72\xa5\xc8\x33\xcf\x88\x18\x46\xc9\xdf\ +\x2f\x40\xaa\xab\x45\xfe\xe9\x9f\x44\x72\x72\xf4\x1d\x04\x22\xb9\ +\xb9\x22\xff\xf0\x0f\x22\x5b\xb7\xfe\xdd\xdd\x45\xf2\x87\x3f\x88\ +\xac\x58\xa1\x15\x59\xd0\x37\xcd\x9c\x39\x22\x4f\x3c\x21\xe2\xf7\ +\xdf\xf5\xf7\x07\x90\x03\x07\x44\xee\xbe\x5b\x5b\x2c\xc3\xfa\xc7\ +\xf8\xf1\x22\xdf\xfb\xde\x19\x2d\x5e\x3e\x75\x5d\x5e\x79\x45\xe4\ +\xca\x2b\x45\x92\x93\xa3\x7a\xca\xf4\xe9\x22\x8f\x3d\xf6\xf7\x05\ +\x14\xa9\xad\xd5\x7e\x83\xcc\xcc\x28\x40\x4a\x4b\xb5\xdf\xa3\xa5\ +\xe5\xef\x5e\x0e\xcb\xa6\x4d\x22\x37\xdf\x2c\x92\x96\x16\x05\xca\ +\xb4\x69\x9a\xa3\xfc\x3d\x88\x1e\x69\x6c\x14\xf9\xf6\xb7\x45\xb2\ +\xb3\xa3\x00\x99\x3c\x59\xe4\xc1\x07\xcf\xfa\x08\x46\xaf\x53\x45\ +\x85\xc8\xad\xb7\x6a\x6b\x67\x18\x28\x73\xe7\x8a\x3c\xf3\xcc\x99\ +\x7d\x13\x89\xcb\xb5\x5d\x7e\xf8\x43\x6d\xd6\x0e\xeb\x20\x65\x65\ +\x22\xbf\xfb\x9d\x88\xcb\xb5\xfd\x2c\x34\x8e\x58\xaf\xca\x4a\x91\ +\xdb\x6e\xd3\x1c\x65\x58\x47\x59\xb1\xe2\xcc\xb6\x7a\xe4\xa1\x87\ +\x44\x4a\x4a\xa2\x00\x99\x30\x41\xe4\x37\xbf\x39\x0b\x90\x4f\x5a\ +\xb3\x3d\x7b\xb4\xe8\x49\x4e\x8e\x5a\x3d\x57\x5d\x75\x66\xfa\x51\ +\xe4\x85\x17\xb4\xa6\x6e\xb5\x46\xad\x98\x9f\xfe\x54\xa4\xad\xed\ +\xac\x2f\xe0\xd3\xd6\x6e\xcb\x16\xad\xcc\x0e\x5b\x3d\x0e\x87\x56\ +\xfa\xcf\x24\xcf\xac\xec\xdc\xa9\xfd\x1e\x71\x71\xfa\x22\x53\x52\ +\xb4\x5e\x72\xe8\xd0\x49\xb9\x48\x19\xea\x10\xf1\x36\x8b\x78\x0e\ +\x89\xb8\xeb\x44\xdc\xb5\x22\xee\x7a\x11\x4f\x83\x88\xaf\x55\x24\ +\x38\xf8\xcc\x69\xbf\x86\xaf\xbf\xae\x45\xcd\xb0\x1f\x25\x35\x55\ +\x2b\xfa\x27\x41\x8f\xb3\x8d\xf9\xc5\xb5\xb7\x0b\x3f\xfa\x91\xce\ +\x3b\x0d\x04\x74\x06\xfa\xa5\x97\xc2\x57\xbf\x8a\x2a\x2e\x56\xb1\ +\x06\x03\xc1\x7e\x08\xf6\x43\x68\x10\x82\x3d\xe0\xef\x81\x83\xbf\ +\x87\x90\x1b\x0c\x1f\x98\x21\x40\x00\x2b\x58\xe3\xc0\xe6\x04\x7b\ +\xea\x6a\x39\xf8\xb0\x10\x9f\x0d\x8e\x74\xb0\xa7\x80\x3d\x0d\x12\ +\x8b\x2e\x56\xd6\x4f\xce\x5a\x3b\x59\xa4\x2e\xbb\x4c\xc9\xe3\x8f\ +\x0b\x7d\x7d\x50\x5d\xad\x13\xba\xff\xf0\x07\x28\x2b\x4b\x1f\xf3\ +\x73\x8f\x39\x48\x7e\xfb\x5b\xe1\xe7\x3f\x87\xa6\x26\xfd\xc2\x92\ +\x25\x70\xff\xfd\xa8\x73\xcf\x8d\xc9\xb9\xc5\x5d\x2f\xf8\x3b\xc1\ +\x73\x08\x06\xab\xf5\xc3\x5d\x03\xbe\x66\x30\xbc\x60\x1a\x11\x50\ +\x44\x1e\x32\xfa\xca\xd5\xa8\x5f\x2c\x60\xb1\x41\x5c\x16\x38\x8b\ +\x21\x65\x32\xa4\x4c\xd5\xcf\x09\x85\x10\x9f\x8b\x4a\xcc\x3f\xe5\ +\x09\x42\xf2\xa3\x1f\x09\x6b\xd6\xe8\xec\x37\x80\xcb\x2e\x83\x1f\ +\xff\x18\xb5\x60\x81\xfa\x5c\x72\x12\xd9\xbc\x59\xf8\x97\x7f\xd1\ +\x29\x7d\x22\x3a\x7b\xec\x9b\xdf\x3c\x61\x80\x88\xaf\x55\xf0\x36\ +\xc3\x40\x15\x54\xdd\x07\xbd\x5b\xc0\x7b\x08\x0c\x3f\x88\x09\x0c\ +\x97\x4b\xc8\xc7\xdf\x1b\x72\x94\xbf\x99\x40\xd8\xab\x01\xd6\xb3\ +\x41\x73\x1b\x47\x3a\xa4\xcd\x84\xac\x65\x48\xf3\x0b\x42\x52\x29\ +\x2a\x7d\xe6\xa9\x03\xcb\x1d\x77\x40\x6d\x2d\xac\x5b\xa7\x13\xb1\ +\xdf\x79\x07\xd6\xad\x43\xda\xdb\x45\xe5\x8f\x0d\x88\xc7\x0c\x24\ +\xe2\xf5\xde\xc7\x77\xbe\x03\xfb\xf6\xe9\x04\xe5\xc4\x44\xf8\xd2\ +\x97\xe0\xbc\xf3\x4e\x80\x6b\xd4\x09\x83\xd5\x50\xfd\x20\x74\x94\ +\x6b\x8e\x61\x0c\x69\x60\x88\x19\x05\x80\x35\x01\xec\xc9\x60\x4b\ +\x02\x5b\x22\x58\xe3\xc1\xe2\x00\x65\x07\x65\x89\x70\x0e\x53\x73\ +\x19\x09\x81\x19\xd0\xc7\x09\xfb\xb4\x58\x0a\xbb\x23\x62\x29\x0c\ +\xfe\x4e\xe8\xec\x86\xae\xf7\xc1\x91\x01\x99\x4b\x91\xea\x87\x84\ +\x8c\x05\x90\x36\xfb\x7e\x65\x77\xde\x7b\x52\xc5\xce\xb8\x71\x4a\ +\xde\x7b\x4f\x27\x82\x0f\x8b\xf0\x3f\xff\x19\x16\x2c\xf8\xfc\x89\ +\x1b\xf9\xcb\x5f\x84\x1f\xfe\x50\x27\x2b\x83\x06\xc7\x2f\x7f\x79\ +\x5c\x6c\x51\xdc\x87\x84\xfe\x3d\xd0\xf6\x2a\xb4\xbf\x01\x43\xed\ +\x7a\x73\x87\x81\x61\x8d\x07\x47\xa6\x16\x15\x71\x99\x90\x38\x0e\ +\x92\x4a\xc0\x39\x0e\xe2\x73\xf5\xeb\xf6\x14\xb0\x26\x46\xc0\xa2\ +\x40\x0c\x30\x02\x10\xf6\x40\xb0\x0f\x02\x3d\x30\xd4\x06\x9e\x06\ +\xf0\xd4\x6b\x70\x04\x7b\xc1\xdf\xad\xf5\x1b\x09\xeb\xe5\x52\x16\ +\x7d\xbe\xf4\x79\x50\x78\x0d\x64\xaf\x80\xcc\x45\x27\x5d\x77\x91\ +\xff\xfc\x4f\xe1\xff\xfd\x3f\x68\x69\xd1\x2f\xdc\x74\x13\xfc\xe8\ +\x47\xa8\x69\xd3\xd4\xe7\x82\x93\x48\x63\xa3\xf0\xad\x6f\xe9\x0b\ +\x10\xd1\x65\x91\x5f\xf9\xca\x31\x03\x44\xfc\x5d\x2e\xfa\x76\xa7\ +\xb3\xff\x97\xd0\xf6\xb2\x06\xc7\xb0\xe2\xa9\x6c\x90\x90\xab\xf5\ +\x85\xe4\x32\xbd\x69\x19\xf3\x20\x65\x2a\xca\x39\xfe\x84\x17\x4a\ +\xfa\xf6\x0a\xfd\x7b\xc1\xb5\x1d\xfa\xab\xc0\xd7\xa8\x41\x14\xf6\ +\x68\x91\xd4\xb3\x09\x5c\x3b\x20\x7d\x01\x8c\x5f\x5d\x2e\xae\xdd\ +\xa8\x8c\x93\x58\x2e\x71\xf3\xcd\xb0\x65\x0b\xbc\xfc\x32\x78\xbd\ +\xf0\xda\x6b\x70\xf1\xc5\x9f\x23\x9d\x64\xdd\x3a\x5d\x41\xe7\xf7\ +\xeb\xea\xb6\xab\xaf\xd6\xb5\x27\xc7\xb4\x49\x7b\x84\xda\x3f\x40\ +\xc3\x93\xe0\x39\xa8\xef\x7a\x44\xdf\xc5\x09\x05\x90\x3c\x09\xb2\ +\x57\x42\xee\x79\x90\xb9\xe8\x6e\x65\x8d\x7f\x38\xa6\x2c\x36\x7d\ +\x96\x3a\x8c\x93\xb9\xb6\x41\xc7\xdb\xd0\x57\x01\x9e\x3a\xcd\x7d\ +\x8c\x21\xe8\xdd\x04\xfd\xbb\x21\xff\x62\xa4\x71\xad\x90\xbd\x02\ +\x95\x58\x30\xe6\x60\x51\x85\x85\x4a\x5e\x7c\x51\xa8\xaa\xd2\x22\ +\x7d\x70\x10\x9e\x7c\x12\xd9\xb5\x4b\xd4\xbc\x79\xea\xb4\x06\x89\ +\xd4\xd7\x0b\x77\xde\x09\x9d\x9d\xfa\x85\x89\x13\x61\xf5\x6a\x54\ +\x49\xc9\x67\xfa\xe2\x12\x1c\x7c\x86\xae\xf7\x57\xb3\xe7\x7f\x6b\ +\x3d\x20\x34\xa0\xc5\x8a\xc5\x01\x89\x45\x90\x36\x07\x0a\x2e\x87\ +\xbc\x0b\x51\xc9\xa5\x27\xe5\xce\x55\xc9\x51\x53\x5d\xba\x3e\x10\ +\x5a\x5f\x81\xee\x0d\x30\xb8\x5f\x9b\xdb\x61\x37\xb4\xbc\xa0\x01\ +\x34\xf1\x56\xa4\x77\x97\xa8\xcc\x79\x63\x0f\x94\x6b\xae\x51\xf2\ +\xad\x6f\xe9\xf2\xd6\xfe\x7e\x5d\xac\xb6\x71\xe3\xe7\x80\x93\xbc\ +\xf0\x02\x1c\x38\xa0\x2b\xf4\xed\x76\xb8\xe1\x06\x5d\x24\xf5\x59\ +\x00\xe2\x39\x24\xd4\x3d\x06\x75\x7f\xd0\x4a\xa9\x19\xd2\x7f\x48\ +\xc8\x87\xb4\xb9\x30\xee\x3a\x28\xb8\x3c\x26\xe2\xe4\xb8\x37\x26\ +\x67\xa5\x02\x90\xee\xcd\x42\xd3\x73\xd0\xf5\x8e\x36\xbb\xc3\x3e\ +\x6d\x86\xef\xff\x15\x0c\x1c\x40\x5a\x5f\x13\x55\x78\xf9\xd8\x7f\ +\xcf\x9b\x6f\x86\x0f\x3f\x8c\x72\xee\x67\x9f\x8d\x39\x37\x89\x29\ +\x48\xa4\xb5\x55\xf8\xc7\x7f\xd4\x6d\x1d\x40\x17\x4d\x5f\x7e\xf9\ +\x67\xaa\xb2\x97\xbe\xbd\x42\xf5\x03\xd0\xf8\x34\xf8\x3b\x34\xf7\ +\xb0\x25\x42\xf2\x54\x28\xba\x06\x26\xdc\x8c\x4a\x9d\x76\xda\x14\ +\x32\xa9\xec\x65\x1a\x2c\x2d\x2f\x0b\x0d\x4f\x42\xf7\x7a\xad\x33\ +\x19\x5e\x68\x79\x1e\x7c\x4d\x48\xfd\x1f\x85\x82\xcb\xfb\x54\xfc\ +\xd8\xf5\x34\x51\xe7\x9c\xa3\xe4\x3b\xdf\xd1\xd6\x4e\x5f\x9f\x06\ +\xcc\x8e\x1d\xa7\x31\x27\x29\x2f\xd7\x2d\x1e\x42\x21\x5d\xad\x76\ +\xf5\xd5\xba\x35\xc3\xa7\x01\xa4\x67\xab\x50\x75\x1f\xb4\xbf\xaa\ +\xd9\x37\xa2\xad\x92\xec\x73\xa0\xec\x0e\x54\xc1\xe5\x0a\x7e\xc4\ +\xe9\x48\xaa\xe8\x2a\x25\xde\x26\x0d\x94\xe6\xe7\xa0\xbf\x52\x7b\ +\x76\x5d\x5b\xa1\xb2\x17\x02\xfd\xe9\xe2\x6d\x11\xe5\x2c\x1a\x3b\ +\x80\xdf\x70\x03\xbc\xf7\x9e\xf6\xc2\xfa\xfd\xf0\xd2\x4b\xc8\x81\ +\x03\x12\xab\xea\xc0\xd8\x82\xe4\xf5\xd7\xa3\x9e\xc0\xfc\x7c\xb8\ +\xf8\xe2\x4f\xe5\x22\xd2\xb5\x49\xa8\xfc\x09\x74\xbe\xa3\x2d\x07\ +\xa5\xb4\xc7\x73\xfc\x6a\x28\xf9\x0a\x2a\xf5\xf4\x2f\x83\x1c\x16\ +\x7f\xd2\xfc\xa2\x50\xf7\x08\x74\x6d\xd0\xba\x94\xbb\x16\xf6\xff\ +\x1c\x8c\x21\xc4\xdb\x24\x63\x25\x26\xd5\x39\xe7\x28\xb9\xf3\x4e\ +\xa1\xb6\x56\x17\xce\xaf\x5f\xaf\x5d\xf7\x31\x22\x4b\xcc\x44\xcd\ +\x86\x0d\x5a\xd3\x1e\x1a\xd2\x2f\x5c\x78\x21\x4c\x9a\xf4\xc9\x9f\ +\xe9\xde\x2c\x54\xfd\x14\x3a\xdf\xd2\xca\x9f\xb2\x42\xea\x4c\x98\ +\xfa\x2f\x30\xe5\xdb\x9f\x0b\x80\x1c\xee\xe8\xba\x46\x31\xeb\x47\ +\x30\xee\x06\x88\xcf\xd1\x80\x1f\x6a\x83\xea\x5f\xc3\xa1\xbf\x20\ +\xde\x31\xcc\xba\xbb\xea\x2a\xdd\xc1\x00\xb4\x12\xfb\xde\x7b\x5a\ +\xfc\x9f\x4e\x20\xa1\xbc\x3c\xca\x45\x9c\x4e\xb8\xe8\xa2\x4f\xb4\ +\x68\xc4\xb5\x4b\xd8\x77\x3f\x74\xbd\xa7\xfd\x0e\xca\xa6\x7d\x1d\ +\x33\xfe\x0d\xca\xee\xba\xf8\x74\x88\x93\x1c\x17\x50\x32\x17\x28\ +\x66\x7c\x1f\x26\x7c\x11\xe2\xf3\x35\x50\xfc\x9d\x50\xf3\x00\x34\ +\xad\x45\x02\x63\x94\x3b\x73\xfe\xf9\x6b\x99\x34\x49\x07\x50\x45\ +\xe0\xed\xb7\x75\x38\xe4\x74\x01\x89\x78\xbd\xf7\xb1\x65\x8b\x46\ +\x30\xc0\x8c\x19\x9f\x68\xd1\xc8\x60\xad\x70\xe0\xd7\xd0\xf9\x6e\ +\x44\xc4\xd8\x20\x63\x01\xcc\xf8\x77\xd4\xc4\x2f\xaa\xd3\x25\xf2\ +\x7a\xdc\x40\x49\x29\x55\x4c\xfb\x2e\x4c\xfc\x72\x14\x28\x43\x1d\ +\x50\xf3\x20\xb4\xbc\xb8\x60\x6c\xce\x99\x72\x13\x2b\x57\xea\x8e\ +\x4b\xa0\x2d\xcc\x18\x89\x9c\xd8\x70\x92\x4d\x9b\xee\xa1\xa9\x49\ +\x2b\xac\x4a\xe9\x0e\x3f\x79\x79\x47\x07\x48\xc0\xb5\x9d\xda\x47\ +\xa0\xfd\x35\xed\xee\x56\x16\x48\x9b\x0d\xd3\xef\x45\x8d\xbf\xfe\ +\x8c\x69\xc3\xa0\x9c\x85\x8a\xa9\xff\x03\xc6\xdf\x04\x71\xd9\xfa\ +\x45\xcf\x21\xa8\x79\x00\x69\x7d\x75\x6c\xc4\xce\x79\xe7\xe9\x75\ +\x57\x4a\x77\x69\xda\xb8\x11\x69\x6a\x92\xd3\x05\x24\xba\xd3\xcf\ +\xb0\xa8\x59\xb2\x84\x8f\x8d\x48\x36\xad\x5d\x40\xf3\x5a\x1d\x13\ +\x41\x20\x69\x12\x4c\xfb\x9f\x67\x14\x40\x46\x80\x92\x18\x01\x4a\ +\xe1\x35\xe0\x48\xd3\xd7\xdb\x57\x01\x07\x1f\x46\x5c\xb1\xaf\x69\ +\x56\x4b\x97\x2a\x8a\x8b\xb5\xc8\x01\xed\xb6\xef\xee\x3e\x4d\x38\ +\x49\x45\x85\x36\xbf\x40\x37\x92\x29\x2e\x3e\x3a\x17\xe9\x5c\x2f\ +\xd4\xfd\x37\x78\x9b\xa2\x66\xee\xa4\xaf\x43\xe1\xd5\x6b\x39\x43\ +\x49\x25\x4d\x54\x4c\xf9\x27\xc8\x5e\xa5\x03\x8c\x62\x6a\x2e\xda\ +\xf4\xac\x4e\x79\x88\x35\x2d\x5a\x04\x69\x69\xfa\xe7\xe1\x8e\x93\ +\xa7\x1a\x24\x52\x55\x25\x34\x37\x6b\xf6\x36\xfc\x25\x33\x33\x3f\ +\xfa\x3e\x5f\x9b\x50\xf7\x07\x9d\x03\x22\x61\xbd\x60\x13\x6e\x81\ +\x71\xd7\xa3\x1c\x29\x37\x71\x06\x93\x4a\x9f\xad\x98\xf4\x75\x9d\ +\x97\x62\xb1\xeb\xd4\x84\xfa\x27\xb4\xa9\x3c\x16\x20\x19\xd6\x4b\ +\xbc\x5e\xa8\xac\x44\x06\x4f\x2c\x3d\xf3\xc4\x39\xc9\xbe\x7d\x5a\ +\x61\x1d\xee\x7f\x3a\x6f\x1e\x6a\xfc\x51\xfc\x01\xcd\xeb\x74\x2c\ +\x26\xec\x01\x94\x0e\xb1\x17\xff\x03\x2a\xe9\xc4\x1b\xe5\xfa\xc3\ +\x72\x57\xab\xdb\x94\x86\x01\x53\xea\xfa\x4d\xa9\xef\x37\xa5\x71\ +\xc0\x94\x36\x8f\x29\x86\x29\xa7\x45\x31\x93\x2a\xbc\x4c\x31\xe1\ +\x4b\x5a\x91\x45\xc1\x50\x2b\xd4\x3f\x81\xb8\x2a\x62\xcb\x4d\x66\ +\xcd\xd2\x4d\x89\x2d\x91\xad\xdd\xbb\x17\xba\xba\x56\x9f\x5a\x67\ +\x5a\x55\x55\x54\xd4\x24\x27\x43\x59\xd9\x47\xb9\x48\x7f\x95\xb0\ +\xed\x9b\xda\x67\x80\xe8\x7c\x8f\xb2\xbb\x50\x99\xf3\x8f\x0b\x20\ +\xcd\x83\xa6\xf4\xfa\x05\x97\x1f\x3a\xbc\xc2\x6f\x2b\xc2\xf4\xfa\ +\xc1\x1d\x14\x82\x86\x4e\x92\x89\xb3\x41\x8a\x43\x91\x95\x40\xdd\ +\xb3\x35\x61\xb2\x13\x14\x19\xf1\x8a\xac\x04\xc8\x4f\xb2\x9c\x1a\ +\xfd\xa7\xe4\xf6\x1d\xf4\xed\x5a\x40\xf3\xf3\xda\x2f\xd4\xf5\x0e\ +\x74\xbc\x85\x84\xfd\x77\x29\x5b\x6c\xa2\xd8\x2a\x2f\x4f\xc9\xad\ +\xb7\x0a\x3b\x76\xe8\x76\xa2\xfb\xf6\x45\xf7\xe7\x94\x81\xe4\xe0\ +\x41\x9d\x46\x07\x1a\x20\x47\x11\x35\x34\x3e\xa3\x23\xa6\x66\x48\ +\x47\x73\x27\xdc\x0c\x59\xcb\x8e\xe9\x34\x2e\xbf\xb9\xbd\x61\x40\ +\x16\x54\xf7\x09\x0f\xec\x32\xd8\xdd\x63\x72\xb0\x4f\xe8\x0f\x80\ +\x11\xe9\xcd\x7b\xe4\x2d\xa9\x88\x34\x3f\x54\x90\x93\xa8\x98\x9a\ +\xae\x98\x9b\xa3\x58\x57\x1b\x96\xd2\x54\xc5\xac\x6c\xeb\x49\x05\ +\x8b\x8a\xcb\x58\x28\x1d\xef\x09\x03\x55\xd0\xb7\x4b\xa7\x5b\x36\ +\x3e\x05\x59\xcb\xd6\x00\xb1\x4b\x75\x98\x36\x4d\xdf\xb0\x3e\x9f\ +\x6e\x23\x3a\x6c\x54\x9c\x0a\x90\x48\x30\x78\x11\x97\x5e\xaa\x53\ +\xe8\x40\xc7\x69\x92\x93\x8f\x70\x9a\x55\x08\x5b\xbf\xae\x33\xbf\ +\x00\x52\xa7\x43\xd1\xf5\x7c\xd6\x58\x46\x97\xcf\x74\x55\xbb\x24\ +\xfd\xe1\x3d\x06\x6f\x35\x99\x54\xf6\x08\xde\x10\x98\xa2\x1f\x72\ +\x04\x28\x0e\x3b\xf7\x68\xa3\x6a\x50\x68\x71\x0b\xef\x34\x43\x6a\ +\x1c\x2c\xca\xb5\xf0\xbb\x5d\x21\x59\x98\x67\x61\x41\xae\xa5\xd4\ +\x6a\x51\x27\xa5\x53\xa3\xca\x3b\x4f\x49\xc5\xf7\x05\x6f\x83\x5e\ +\x93\xbe\x0a\xe8\x7c\x17\x09\xb8\xb6\xab\xb8\x8c\x85\x31\x39\xc9\ +\xf0\x3e\x74\x76\x6a\xbd\xa4\xad\xed\x14\x72\x92\x9a\x9a\x72\x06\ +\x06\xa2\x9d\x97\x8b\x8b\xb5\x09\x3c\x9a\x5a\x5e\x00\x6f\x9d\x4e\ +\x17\xb4\xc6\xc3\xf8\x5b\x20\x6d\xd6\x67\xb2\x66\x76\x74\x18\xf2\ +\xe8\x5e\x83\xbf\xd5\x9a\xd4\xf6\x0b\x7e\x43\x03\x03\xc0\x66\x81\ +\xf4\x38\x48\x8d\x53\x24\xdb\x21\xd1\x06\x0e\xab\xc2\x66\xd1\xe0\ +\x30\x4c\xfd\x7e\x5f\x18\xdc\x01\xe8\x0b\x68\x70\x85\x4c\xe8\x1d\ +\x82\x37\x1b\x4d\xde\x6f\x81\x05\xb9\x26\x37\x4c\xb2\xd6\x55\xf5\ +\x18\xcc\xc8\x3a\x49\x9c\x65\xc2\xcd\xd0\xb5\x1e\x7a\x36\x6b\x25\ +\xbe\x65\x1d\xe4\x5d\x1c\x3b\x27\xdb\xe8\x7d\x30\x0c\x68\x6a\x42\ +\x06\x07\x9f\x51\x29\xc7\x67\x20\x9c\x18\x48\x86\x7b\x94\x0e\xd3\ +\xf8\xf1\x87\xf5\x52\x17\x77\xbd\xf0\xe1\x3f\x42\x20\x52\x3f\x94\ +\x3a\x03\xf2\x2e\xf8\x54\x6b\xa6\xc5\x6d\xc8\x86\x16\xe1\x07\x9b\ +\x0c\xb6\x77\x9a\x78\x22\x9c\xc3\xa2\xd0\x3a\x85\x53\x31\x2e\x59\ +\x8b\x8f\x49\xe9\x8a\xf1\x29\x8a\xdc\x44\x48\xb6\x2b\x1c\xd6\x88\ +\x32\x6b\xc0\x60\x40\xe8\xf0\x09\x87\x06\xe0\x80\x4b\x03\xad\xd5\ +\x03\x6d\x5e\xc1\x1d\xd4\x00\xda\xd8\x26\xec\xe9\x0e\x73\xd9\x44\ +\x0b\xaf\xd6\x87\xe5\x82\xf1\xd6\xbb\xe3\x6d\xea\xe1\x31\xe5\x26\ +\xe9\xb3\x95\xec\xf9\xa1\x4e\xea\x0e\x74\x41\xff\x1e\xe8\xdd\x1a\ +\xbb\x13\x8c\x1f\xdf\x47\x62\x62\xb4\x1e\xa7\xb5\x15\xfa\xfb\x57\ +\x9f\x1a\x4e\xd2\xd1\xa1\x43\xd3\x44\x84\xff\x91\x5e\xd6\x8e\x77\ +\x74\xa9\x83\x84\xb5\xeb\xbd\xf0\x6a\x48\x9a\xd8\xf7\x49\x87\xdc\ +\xdf\x6b\xc8\x23\x7b\x0c\x9e\xaa\x16\x5a\xbd\x42\xd8\xd4\xe0\xc8\ +\x73\x42\x59\xaa\x62\x69\xbe\x85\x55\x45\x8a\xb9\x39\x96\xbe\x9c\ +\x44\xcb\x31\xe5\x69\x34\x0d\x9a\xb2\xa3\xd3\xe4\xbd\x16\x93\x5d\ +\x5d\xc2\xc1\x7e\x61\x20\x00\x03\x41\x78\xae\xd6\xa4\xaa\x57\x68\ +\x18\x94\x35\x8d\x83\xe6\x9a\x09\x29\x63\xac\xdc\x16\x5e\x0b\x6d\ +\xaf\xe9\x02\x32\x33\x08\xed\x6f\x20\x03\xfb\x25\x16\x39\x33\x2a\ +\x27\x27\x43\x2e\xbd\x54\xb0\x5a\x35\x27\x19\xbd\x4f\x27\x1d\x24\ +\xdd\xdd\x51\x7d\x24\x21\x21\xea\xc4\x19\x51\x28\xde\x8d\x78\x56\ +\xd1\x09\xcb\xd9\xab\xf8\xa4\x04\x9c\xed\x1d\x86\xfc\x62\xbb\xc1\ +\xeb\x0d\x26\x7d\x7e\x2d\x36\x52\x1c\x30\x2d\x43\x71\xe9\x44\x0b\ +\x57\x97\x58\x4e\x48\xd9\x1c\x3f\x6a\xe3\x3f\x68\x35\xe4\x85\x5a\ +\x83\xf5\xad\xc2\xc1\x3e\xc1\x17\x86\x7d\x2e\xe1\x67\x5b\x0d\x3a\ +\xbc\x50\xed\x32\x65\x4a\xc6\xd8\x01\x45\x65\xce\x57\xb2\xf5\xeb\ +\xc2\xe0\x01\x1d\x9e\xe8\xfd\x10\xdc\x75\xb1\x3b\x41\x56\x96\xf6\ +\xbc\x0e\x0d\xe9\x7d\x3a\x65\x20\xe9\xef\xd7\xf1\x1a\xd0\x03\x82\ +\xe2\xe2\x0e\x57\x58\x3f\xfc\x47\x9d\x2c\x0c\x90\xb3\x0a\x92\x26\ +\x7e\xec\xa1\x3e\x6c\x37\xe4\xbe\x6d\x06\xef\xb7\x98\x78\x82\x9a\ +\x31\x8d\x4b\x52\x5c\x30\x5e\x71\xfb\x34\x2b\xcb\x0b\xad\xea\xdf\ +\x63\xb8\x49\x2b\x0b\x35\xd8\x5e\xac\x35\xe4\xaf\x07\x0c\x36\xb6\ +\x99\xf4\xfa\xa1\x67\x08\xfe\x6b\xb7\x81\x37\x24\xec\xeb\x35\x64\ +\x7a\xe6\x18\xea\x29\x79\x17\x43\xc7\x5b\x1a\x24\x81\x5e\x70\x6d\ +\x8f\x9d\x02\x9b\x9e\x1e\x05\xc9\xe8\x7d\x3a\xe9\x20\x19\x1c\x8c\ +\x9e\x3c\x25\x25\x1a\x33\x00\xad\x94\x05\x22\xf1\x19\x8b\x03\x72\ +\x56\xa1\x92\x8f\x9e\x3a\xb0\xa3\xd3\x90\x9f\x6d\x35\x78\xb7\xd9\ +\xc4\x1b\xd2\x4a\xe9\xd4\x74\xc5\x6d\xd3\x2d\xdc\x30\xc9\xca\xb8\ +\xe4\xb1\xbb\xa3\xaf\x29\xb3\xaa\x1a\x97\x29\x8f\x55\x86\x59\x57\ +\x67\xd2\xe2\x01\x77\x08\x1e\xdf\x67\xa2\x14\x1c\xec\x33\x65\x52\ +\xfa\x18\x9d\x3f\x6b\xa9\xf6\x19\x79\x22\x22\xb9\x67\x13\x8c\xbf\ +\x29\x36\x0a\x6c\x4a\x8a\xce\x31\x3e\x72\x9f\x4e\xba\xc7\x75\x68\ +\x28\x3a\x21\xc2\xe9\xd4\x29\x8b\xc3\xd4\xbb\x5d\x97\x1d\x00\x24\ +\x14\x69\xd3\xf7\x28\x54\xdb\x67\xc8\x83\xbb\x0c\xd6\xb7\x44\x01\ +\x32\x37\x5b\xf1\xbd\x45\x56\xbe\x35\xd7\x56\x3a\x96\x00\x19\xb1\ +\x18\x33\x2c\xea\x07\x4b\x6d\xf7\x7f\x75\x96\x95\x92\x54\x6d\x21\ +\x79\x43\xf0\xc4\x3e\x93\xa7\xab\x0d\x5a\xdd\xe6\x98\x44\x6d\x55\ +\x62\x81\x22\x6d\x36\xd8\x93\xf4\x0b\x7d\x7b\x75\x7e\x6f\x2c\x68\ +\xf4\x7e\xf8\x7c\xba\x8a\xf2\x94\x80\x64\x78\x4c\x19\x68\x2e\x12\ +\xf9\x52\xe2\xef\x72\xe1\xa9\xd5\xb9\x9e\x00\xe9\x73\x75\xa6\xd6\ +\x91\x1f\x37\xe4\xa2\xc7\x2a\x0d\xde\x6e\x32\x19\x0c\x6a\xa7\xd7\ +\x9c\x2c\xc5\x77\x17\x5a\xb9\x69\x8a\x4d\x9d\x2c\xdf\x05\x80\xd3\ +\x61\xb9\xf7\x5f\x17\xda\xd5\x1d\x33\x2d\x14\xa7\x28\xac\x0a\x06\ +\x83\xf0\x58\xa5\x41\x79\xa3\x39\x76\x27\xce\x5c\xa8\xeb\x8d\x41\ +\x5b\x3a\x9e\x18\x5d\xb2\xc3\x11\x75\xcd\x1b\xc6\xd1\xa7\x7f\x9d\ +\x14\x90\x18\x46\xb4\xf0\xda\x6a\x8d\xce\xae\x1b\x3c\x98\x4e\xd0\ +\x15\x2d\xc3\x4c\x9f\x03\x71\x99\x1f\x49\xe1\x5e\x5b\x13\x2e\x7f\ +\xa1\x4e\xe8\x89\xa8\x2d\x65\x69\x8a\x6f\xcf\xb7\x72\x6d\x99\xed\ +\x94\xa5\x0d\xfc\x8f\xf9\x76\xf5\xa5\x69\x16\xf2\x9d\xda\xaa\x6a\ +\xf7\xc2\xa3\x95\x06\xeb\x9b\xc3\x63\x93\x03\x92\x36\x3b\x0a\x12\ +\x31\x60\x60\xbf\x0e\x86\x9e\x28\xd9\x6c\xd1\xfd\x10\x39\x85\x20\ +\x19\x3d\xd0\x70\x74\x95\xbe\xfb\xa0\x2e\xbc\x06\x5d\xa4\x9d\x32\ +\x99\x23\x95\xb1\x7d\xbd\x86\xfc\x79\x9f\x49\x93\x5b\x10\x20\x27\ +\x01\xee\x9c\x65\xe1\x8a\x12\xcb\xfd\x9c\x62\xba\x7b\xb6\x75\xc7\ +\x95\xc5\x16\x52\x1d\xda\xc2\xda\xd9\x25\x3c\x7b\xd0\xa4\x61\x20\ +\xf6\x62\x47\xa5\xcf\x56\x38\xb2\xb5\x8b\x60\x78\xed\xc2\xee\x13\ +\x3f\xf0\xd1\x66\x0d\x9e\x12\x90\x8c\x46\x6b\x28\x14\x15\x3d\x9e\ +\x06\x9d\xb7\x0a\x3a\xd9\x26\x3e\xff\x23\x1f\x7d\xfa\x80\xc9\xde\ +\x5e\x21\x64\x82\xc3\x0a\xd7\x96\x59\xb8\xb2\xc4\x82\xd3\x6e\xb9\ +\xf7\x54\x83\x24\x23\xde\xb2\xf0\xab\xb3\xac\x2c\xc8\xb5\x10\x67\ +\xd5\xb1\xa1\x17\xea\x4c\xb6\x77\x8c\x91\xd8\x49\x2c\xd4\xde\x68\ +\x00\x6f\x03\x84\xbc\x27\x7e\xcc\xd1\xfb\x31\x3c\x4c\xf2\x94\x80\ +\x24\x2e\x2e\xaa\x1c\x8d\x1e\x73\x36\xd4\xaa\x83\x57\x00\x09\x79\ +\xe0\x48\x39\xdc\x9a\xe9\x30\xe4\xad\x26\x13\x57\xe4\x2d\xd3\x33\ +\x14\xb7\x4c\xb1\x52\x9c\x6a\x3d\x6d\xb2\xd3\x66\x65\x5b\xd5\x2d\ +\x53\x2c\x14\x25\x29\x14\xd0\xe5\x83\x75\x75\x26\x07\xfb\xc6\x40\ +\x89\x4d\x1c\xa7\xdb\x65\x80\x4e\x9a\x0e\x7b\x4e\xfc\x98\xc3\x93\ +\xc5\x20\x3a\x80\xf2\x94\x80\xc4\xe9\x8c\x22\xd4\xe3\x89\x6a\xd0\ +\x81\x1e\x9d\x58\x03\x10\x9f\x17\xbd\x4b\x22\xf4\x62\xbd\x49\xa3\ +\x5b\x30\x05\x12\x6c\x70\xcb\x14\x0b\x0b\x72\x2d\x77\x73\x9a\xd1\ +\x4d\x53\xac\x17\x2f\xce\x53\x24\xda\xb5\xd8\x79\xb7\xd9\xa4\xb2\ +\x67\x0c\xb8\x49\xc2\xa8\x35\x0a\x0d\x42\x28\x06\x20\x19\x1e\x5c\ +\x09\x3a\xd8\x77\x02\x9c\xe4\xc4\xfc\x24\xa9\xa9\x51\x5b\x7c\x78\ +\x28\xf3\xf0\x85\x4a\xe4\x0b\xc6\x65\x68\x3f\x49\x84\xea\xfb\x4d\ +\xb9\xb3\x3c\x44\x5f\x84\x8b\xcc\xca\x54\x9c\x5b\x64\xe1\xd3\xe2\ +\x25\xde\xb0\xf7\xbe\x66\x7f\xf3\x3d\x03\xe1\x01\x3c\x86\x87\x21\ +\x63\x08\x13\x93\x38\x15\x87\xd3\xe6\x24\xc5\x96\x42\xa6\x3d\x93\ +\xc2\xf8\xd8\x0d\x6e\x76\x58\xd5\x5b\x2f\xd4\x86\xd9\xd1\xa5\xbd\ +\xb2\x2e\xbf\x06\x4a\xf3\xa0\x29\xe3\x62\xe9\xb6\x77\x8c\x5a\x23\ +\xc3\xaf\x4b\x45\x4f\x94\x46\x3b\xd0\x46\xfb\x4c\x4e\x3a\x48\x32\ +\x32\x0e\x77\xd8\xf8\x7c\x88\x69\x94\xf0\xd6\xaa\xa8\x65\x63\x4b\ +\xd6\xca\x6b\x84\x36\xb4\x6a\x87\x95\x21\xda\xe4\xbd\xaa\xc4\xc2\ +\xe4\x74\xf5\xb1\xca\x6a\xe3\x50\xa3\xd4\xfa\x6a\xf9\xaf\x96\xff\ +\x62\xe7\xe0\x4e\x6a\x7d\xb5\x74\x04\x3b\x08\x44\x38\x95\x55\x59\ +\xc9\xb4\x67\x32\x21\x7e\x02\xb3\x93\x67\xf3\x7c\xe7\xf3\x32\xd5\ +\x39\x95\xe9\x49\xd3\x63\xb2\x89\x97\x17\x5b\x2f\xfe\xeb\x01\xb3\ +\xbc\x69\x50\x47\x95\xdf\x6f\x11\xbe\x34\x35\xc6\x12\xc7\x9e\xac\ +\xd3\x1a\xb5\xc6\x19\xd5\xe7\x4e\x84\x5c\xae\xe8\x4d\x3b\xec\x7d\ +\x3d\x25\x20\xc9\xcd\xd5\x63\x56\x87\x15\xa5\x9e\x1e\x18\x68\xab\ +\x8b\xb6\xa6\x02\x2c\x71\x91\x16\x54\x9a\x36\xb7\x99\xf4\x0e\xe9\ +\x45\xce\x4d\x84\xa5\x05\x16\x9c\x8e\x8f\x2a\xab\x7e\xc3\x7f\xd7\ +\xa6\xfe\x4d\x6b\x7e\xd3\xf4\x1b\xca\x7b\xcb\x69\x0b\xb4\x11\x92\ +\x10\x22\xc2\xf0\xbf\x61\x1a\x0c\x0f\xd2\xe8\x6f\xe4\x83\xfe\x0f\ +\x48\xb1\xa5\xb0\x22\x6d\x05\x4f\xb6\x3f\x29\xcb\xd3\x96\x33\x21\ +\xe1\xc4\xd2\x23\x1d\x56\xf5\xd6\x43\xbb\x42\x6c\xeb\x80\x36\x2f\ +\xd4\x0f\x68\xae\x12\x53\x3a\x62\x8d\x30\x83\x27\x7e\xcc\xde\xde\ +\x28\x48\xf2\xf2\xa2\xfb\x74\xd2\x41\x72\xe4\xc9\xdb\xda\x60\xe8\ +\x88\xbb\x60\xa4\x47\x19\xb4\xba\x0d\xb9\xfd\xf5\x30\xbe\x88\x24\ +\x5a\x94\xa7\xc8\x77\x7e\x74\x0f\x9b\x86\x9a\xe4\xd1\xd6\x47\x79\ +\xa2\xed\x09\x0e\x0e\x1d\x24\x68\x06\x11\x04\x85\xc2\x69\x75\x92\ +\x64\x4d\x22\xce\x12\x87\x52\x8a\x90\x19\xc2\x6b\x78\xf1\x18\x1e\ +\x42\x12\xc2\x15\x72\xf1\x4a\xcf\x2b\x6c\x19\xd8\xc2\xea\xdc\xd5\ +\xec\x18\xd8\x21\x0b\x52\x4f\xac\x33\xe1\xf2\x02\x0b\x7f\xde\x6f\ +\xd2\xee\xd5\xd6\xd8\xae\x6e\xa1\xd5\x6d\x4a\x61\xac\xbc\xc1\xca\ +\xca\x61\x29\x53\x72\x62\x7a\x8f\x1c\x3a\x24\xdc\x74\x53\xd4\x0c\ +\xce\xcf\x3f\x85\x20\x29\x2c\xd4\xca\xab\x8a\x74\x33\x6c\x68\x00\ +\x5f\xe0\x88\x0b\x36\xd0\x6d\x0d\xa1\xa6\x0f\x5c\xfe\x68\xe2\xd0\ +\xc2\x5c\x0b\xe3\x92\xd5\x61\x0a\x6b\xbd\xb7\x5e\xd6\xb4\xac\xe1\ +\xa9\x8e\xa7\xe8\x0a\x76\x61\x62\xe2\x50\x0e\xf2\xe3\xf2\x19\x17\ +\x3f\x8e\xb2\xc4\x32\x26\xc6\x4f\x24\xdd\x9e\x8e\x05\x0b\x6e\xc3\ +\x4d\x5b\xa0\x8d\x5a\x5f\x2d\x8d\xfe\x46\x9a\xfd\xcd\x78\x0d\x2f\ +\x5d\xc1\x2e\x1e\x6e\x79\x98\x16\x7f\x0b\x1f\xb8\x3e\x90\x95\x19\ +\x2b\x8f\x7b\x43\xe7\xe7\x5a\xd5\x0d\x2f\x06\x64\x5f\x2f\x04\x4d\ +\xa8\xec\x31\x19\x08\xc4\xae\x42\x16\x09\x1f\x0e\x0c\x65\x3d\xb1\ +\xe3\x35\x34\xe8\x8c\xb4\x61\xff\x48\x51\x11\xe4\xe5\x95\x9e\x12\ +\x90\xa8\xe2\x62\x25\x57\x5c\xa1\xf3\x16\xc2\x61\xdd\x3a\x72\x28\ +\x74\xf8\x45\x0e\x77\x47\x04\x6a\xfb\x05\x77\x44\x97\xb2\x5b\x60\ +\x4a\xba\x3a\x4c\x61\x6d\x19\x6a\x91\x07\x9b\x1e\xe4\xe9\xce\xa7\ +\xe9\x09\xf5\x20\x08\x39\x8e\x1c\xe6\x26\xcf\xe5\xb2\xcc\xcb\xb8\ +\x20\xe3\x02\x26\x39\x27\x1d\x75\xb3\x3b\xfc\x1d\xb2\x75\x70\x2b\ +\xaf\xf4\xbc\xc2\xa6\xfe\x4d\x34\xfb\x9b\x09\x4a\x90\x97\x7a\x5e\ +\x22\x24\x21\x3e\xec\xff\x50\x96\xa6\x2d\x3d\x6e\xa0\x94\xa6\x2a\ +\x36\xd9\x85\x60\x40\x8b\x9c\x81\x60\xec\x30\xa2\x23\xe5\xa3\x3c\ +\xa2\xd6\xf8\x13\x3b\x5e\x5d\x5d\x14\x24\x36\x1b\x8c\x1b\xc7\x89\ +\x0c\xd2\x3e\xf1\x44\xe8\x71\xe3\x74\x2e\x89\xdb\x1d\x49\x8a\xf6\ +\x6b\x9b\x5f\x59\x35\x17\x09\x0d\x8c\x74\x2c\x6a\x76\x0b\x43\x21\ +\xcd\x46\x32\xe3\x21\x3b\xe1\xf0\x3d\x7b\xaa\xe3\x29\x9e\xef\x7e\ +\x9e\x9e\x90\xce\x87\x9d\x10\x3f\x81\x2f\xe4\x7c\x81\x5b\xf3\x6f\ +\x65\x4a\xd2\x94\x4f\xdc\xe0\xbc\x78\x9d\x11\xe7\x0d\x7b\xef\x7b\ +\xb2\xe3\xc9\x7b\xfe\xdc\xfe\x67\xf6\x7a\xf6\xe2\x37\xfd\xbc\xd1\ +\xfb\x06\x19\xf6\x0c\x6a\xbc\x35\x32\xd9\x39\xf9\xb8\x80\x52\x92\ +\xaa\x70\xda\x15\xfd\x01\x1d\x46\x18\x08\xc4\x50\x2f\x19\xb5\x46\ +\x28\x8b\xee\x52\x7d\x22\x54\x5d\xad\xf7\x63\xd8\xb8\xf8\x98\x92\ +\xdb\x93\xe3\x27\x01\x9d\x99\x9d\x12\x71\x96\x35\x37\x43\x67\x17\ +\x28\x67\x54\x5b\x0f\xf4\x8e\xf8\x4c\xba\x7d\x42\x20\x72\xc3\xe4\ +\x3a\x15\x09\xa3\xac\xb2\x37\xbb\xdf\x94\xb5\x9d\x6b\xe9\x08\x74\ +\x20\x08\xe3\xe2\xc7\x71\x77\xd1\xdd\x7c\x73\xdc\x37\x3f\x15\x20\ +\x87\xb9\x6e\x6c\xce\x7b\xbf\x5a\xf4\x55\xf5\x9d\x09\xdf\x61\x6e\ +\xf2\x5c\xe2\x2c\x71\x84\x25\xcc\xba\xae\x75\x94\xf7\x96\x13\x34\ +\x8f\x6f\xa6\x5f\xbe\x53\x11\x6f\x1b\x56\xaa\xa1\x3f\x10\x43\x4e\ +\xe2\xef\x01\x23\xc2\x9a\xac\x89\xb1\x05\xc9\xe4\xc9\xda\xba\x39\ +\xa5\x20\x99\x39\x53\xfb\x4b\x40\x6b\xd3\xfb\x0e\x40\x30\x5e\x6b\ +\xec\xa0\x5b\x44\x85\x75\x34\x78\x20\xa8\x13\x91\x01\xd2\xe2\x74\ +\xe2\xf2\xc8\xdd\xdf\xf9\x24\xf5\xfe\x7a\x4c\x4c\x52\x6d\xa9\xdc\ +\x96\x77\x1b\xab\x73\x57\x8f\x70\x88\x63\xa5\x6b\x73\xae\x55\x77\ +\x16\xde\x49\x49\x42\x09\x56\x65\xc5\x6d\xb8\x79\xb2\xe3\x49\x76\ +\x0e\xee\x2c\x3f\x9e\xe3\xa5\xc7\x2b\x1c\xa3\x56\x6b\x30\x96\xe2\ +\xc6\xdf\x1e\x4d\xce\x72\x64\x68\xb7\xc1\xf1\xaa\x37\xd5\xd5\x42\ +\x5b\x5b\xd4\x47\x32\x73\x66\xf4\x26\x3e\x65\x20\x99\x32\xe5\xf0\ +\x8a\xb1\xed\xdb\xc1\xef\x3c\xdc\xcd\x1c\xd4\x2d\x29\x42\x66\x54\ +\x69\x4d\xb0\xe9\x70\x3c\xc0\xdb\xae\xb7\xef\xd9\xe5\xde\x85\xcf\ +\xf0\xa1\x50\x9c\x9f\x7e\x3e\x37\xe4\xde\x40\x41\xfc\x89\xb5\xba\ +\xbc\x25\xff\x16\x75\x45\xd6\x15\xa4\xd9\x74\x5a\x65\xa5\xa7\x92\ +\xcd\xfd\x9b\x8f\x8b\x9b\xc4\xdb\x60\x74\xd0\x20\x10\x8e\x21\x48\ +\xbc\xcd\x51\x90\x24\xe4\x83\xfd\x04\x38\xc9\x8e\x1d\xda\x47\x02\ +\xda\x15\x3f\x6b\xd6\xd1\x2b\x2a\x4f\x26\x48\xd4\xb8\x71\x8a\xb2\ +\x32\xdd\x16\x1c\x74\x17\x40\x5f\x12\x58\x22\xbf\x87\x3d\xba\x57\ +\x3b\x1f\xdf\xe9\xfd\xed\xde\xb7\xe9\x08\xe8\x64\x9b\x4c\x7b\x26\ +\x37\xe6\xde\xc8\xd4\xa4\xd8\x74\x39\xba\x29\xf7\x26\xca\x12\xcb\ +\xb0\x2a\x2b\x21\x09\xf1\xb6\xeb\x6d\xea\x7c\x75\xe5\x9c\x26\x24\ +\xfe\x2e\x17\x43\xad\xd1\x30\x46\x52\x89\x6e\x77\x7e\xbc\xb4\x75\ +\x6b\x14\x24\x99\x99\x30\x75\xea\x89\xbb\x71\x62\x72\xa5\x8b\x17\ +\x47\xe5\x5e\x73\x33\xb4\x84\x21\x64\x8d\x9a\xc0\x83\xfb\x31\xbd\ +\x6d\xe2\xb0\xe8\x1c\x0d\x00\x7f\x58\xc7\x6e\x5a\x87\x5a\xa5\xca\ +\x5b\x85\xc7\xd0\xf1\x8a\xc5\x29\x8b\x99\xee\x9c\x1e\xb3\x4d\x98\ +\x99\x3c\x53\x2d\x4c\x5e\x48\xb2\x55\xb3\xf0\xdd\xee\xdd\xb4\xf9\ +\x8f\xbd\x58\xc9\x1f\xd6\x5e\xe2\x61\x8a\x8b\x55\xb7\xb9\x23\x73\ +\x6f\x92\x27\x83\x23\xfd\xb8\xda\x27\x4a\x7b\xbb\xb0\x67\x4f\x54\ +\x1f\x99\x33\xe7\x84\x95\xd6\xd8\x81\x64\xe9\x52\xed\x7d\x55\x4a\ +\x9b\xc2\x3b\x9b\xc1\xad\xa2\x39\x12\xae\x0a\x54\xa8\x8f\x14\x87\ +\x4e\x4f\x04\xad\xf8\x05\x0c\xa1\xc6\x57\x43\x5f\xb8\x6f\xc4\x59\ +\xb6\x34\x6d\x29\x13\x12\x26\xc4\x34\xd8\xb7\x34\x75\x29\x19\x76\ +\x9d\xa4\xdf\x1f\xee\xa7\xc1\xdf\x70\xcc\xc7\xe8\x0f\xc8\x88\x3e\ +\x05\x3a\x8b\x3f\x26\xd4\xb7\x2b\x9a\xe6\x69\x89\xd3\x6d\xcf\x8f\ +\x37\x11\x7a\xcb\x16\xdd\xaa\xdd\x30\xf4\x5e\xac\x58\x01\x79\x79\ +\x3b\x4e\x0b\x90\xa8\xb9\x73\x15\x93\x27\x6b\x53\x18\x60\xcb\x6e\ +\x18\x48\x06\x15\xf9\x7d\x60\x2f\xf8\x5a\xc9\x8e\x37\x88\x8f\x30\ +\x98\x4e\x9f\x30\x14\x86\x66\x7f\x33\xbe\x48\x9a\xa3\x5d\xd9\x29\ +\x4b\x28\x23\x3e\xc6\x2d\xc0\xa7\x3a\xa7\x92\x1c\x51\x06\x4d\x4c\ +\x5a\x02\x2d\x74\x05\x8e\x6d\xe2\x54\xbb\x57\x18\x8a\x24\xa7\xc5\ +\x59\xb5\xe2\x1d\x13\x72\xed\x18\xd1\xd9\x48\x2c\x02\xe7\x84\xe3\ +\x3f\xd6\xbb\xef\x46\x7b\xe8\xa6\xa5\xc1\xe2\xc5\xa8\x8c\x13\xcf\ +\xbc\x8f\x9d\xdb\xf0\xdc\x73\xb5\x02\x0b\xd0\xd4\x0c\x75\x0a\xfc\ +\xc3\x36\x63\x37\xf4\xef\xa6\x28\xde\x4d\x42\x24\x33\xb1\x67\x08\ +\x7a\x87\x84\xae\x40\x1f\xc1\x48\xac\x22\xde\x12\x4f\xba\x3d\xf6\ +\x83\xa0\x4a\x12\x4b\xee\x8e\xb7\x44\x1d\x54\xfd\xa1\x7e\x82\x66\ +\xf0\x98\x4e\x74\x68\x40\xf0\x45\x0c\x86\xac\x04\xdd\xb1\xe0\x84\ +\xf5\x91\xc1\x6a\x61\x70\x7f\x34\x17\x38\x63\xc1\x51\x73\x81\x3f\ +\xb3\x55\xb3\x6d\x9b\x0e\xb4\x82\xee\x53\x32\x71\x62\x4c\xd6\x2f\ +\x76\x20\x39\xff\x7c\x18\x3f\x9e\x91\xaa\xb1\x6d\x3d\xe0\x32\x19\ +\x99\x2d\xd3\xf9\x1e\x65\xf1\x5d\x24\xd9\x65\xc4\xd2\xa9\xee\x13\ +\xfa\x02\xa1\x91\x60\x9d\x45\x59\xb0\xa9\xd8\x77\x32\x8f\xb7\xc6\ +\x3f\x6c\x19\x75\xa9\x21\x09\x61\x72\x6c\xf1\x91\xba\x7e\xc1\x1b\ +\xb1\x68\x8a\x53\x15\xa9\x71\x31\xd0\xab\x3b\xdf\xd7\xed\x38\xc4\ +\xd4\xa2\x39\x6b\x39\x24\x15\x1f\x9f\xfb\xfc\xcd\x37\xf5\x74\x32\ +\xd3\xd4\x5e\xd6\x4b\x2e\x81\xb2\xb2\x8b\x4f\x2b\x90\xa8\xc9\x93\ +\x15\x8b\x17\x47\x6d\xf2\xca\x0e\x68\x54\x10\x8c\xc8\x97\xde\x0f\ +\x99\x6c\xad\x21\xc3\x11\x1c\x51\x5e\x77\x74\x0a\xfe\x50\x3c\xd6\ +\x88\x1b\x3f\x6c\x86\x47\x52\x00\x62\x49\x1d\xfe\x0e\x09\x4b\xd4\ +\x66\x8d\xb7\x44\xcf\xf9\x59\x68\x77\x97\x21\x2d\x1e\x08\x45\x1c\ +\x81\x33\x33\x15\xa9\xb1\x10\x37\x9d\xef\x68\x47\xda\xb0\xa8\xc9\ +\x98\x87\xb2\x1c\xbb\xfb\x5c\xda\xdb\x85\x37\xde\x88\xf6\x47\x9b\ +\x30\x01\x96\x2e\x45\x39\x62\xd3\xc5\xd2\x12\xd3\xdd\xb8\xea\x2a\ +\xed\xa6\x57\x0a\x7c\x7e\xd8\x1e\x02\x57\xc4\xf6\x0d\xb8\xc8\x73\ +\xbf\x4d\x59\x7c\x37\x09\x11\x66\xb1\xbd\x53\x08\x04\x33\xb1\xa2\ +\xb5\xc0\x80\x04\xe8\x0a\x76\xc5\x1c\x24\xad\x81\x56\x86\xcc\x68\ +\x61\x7b\x96\x3d\x8b\x34\x5b\xda\x67\x4e\xb8\xde\xd8\x66\xd2\x3d\ +\xa4\x2f\xc3\x61\x81\xb9\x39\x8a\xa2\x13\x8c\x00\x4b\x5b\xb9\x1e\ +\xf4\x64\xf8\x34\xb7\xcd\x3d\x17\x9c\xe3\x8f\xef\x60\x6f\xbf\xad\ +\x9b\xd5\x04\x02\x7a\xed\xaf\xbc\x32\x66\xa2\x26\xe6\x20\x51\x17\ +\x5c\xa0\x98\x37\x2f\xda\xf6\x60\xaf\x1f\x0e\x99\x10\xd4\x5e\x12\ +\x5b\xc7\x6b\x2c\x4f\xaa\x21\xd3\xa1\x85\x7b\x8b\x47\xe8\x1d\x9c\ +\x88\x61\x38\x47\xc4\xc0\x7e\xcf\xfe\x63\x56\x2a\x3f\x8d\xf6\x7a\ +\xf6\x32\x10\xd6\xdd\x7e\x6c\xca\xc6\xf8\xf8\xf1\x38\x6d\x9f\x6d\ +\x2c\x5a\xd0\x90\x8b\x36\xb4\x9a\xf4\xfa\xb5\x48\x9c\x90\x02\x53\ +\xd2\x63\xb0\x6c\x2d\x7f\x03\x5f\xa4\x19\xaf\x3d\x05\xf2\x2e\x41\ +\x25\x1d\xfb\x74\x53\xe9\xea\x72\xb1\x6e\x1d\xb4\xb7\xeb\x17\x72\ +\x72\xe0\xd2\x4b\x51\x45\xb1\xeb\x65\x6f\x89\xf9\x6d\x7b\xfd\xf5\ +\x51\x6e\xe2\x36\x60\xab\x40\x6f\x84\x9b\x78\x1a\x58\xc1\xeb\x8c\ +\xb3\x77\x63\x55\xda\xfb\x5a\xd3\x59\x4c\xc0\x9f\x8d\x05\xcd\xfe\ +\x37\x0e\x6c\xa4\x2d\xd0\x16\x53\xed\x75\xf3\xc0\x66\x5c\x21\x8d\ +\xbb\x3c\x47\x1e\x13\x12\x3e\xbb\x05\x51\xde\x68\x94\x57\xf5\x6a\ +\x3f\x89\x02\x56\x15\x59\x28\x4c\x3a\xb1\xf5\x97\x8e\x77\x44\xf7\ +\x9f\x8f\x28\x99\x59\xcb\x74\x01\xdb\xf1\xd0\xeb\xaf\xa7\xb3\x73\ +\xa7\xae\xa6\x54\x0a\x2e\xbf\x5c\xc7\xd3\x62\x48\x31\x07\x89\xba\ +\xee\x3a\xc5\xa2\x45\x51\x6e\x52\x29\x50\x0d\xf8\xb5\x01\x5a\x3c\ +\xb8\x8e\x25\x71\x15\xa4\xda\x74\x92\xeb\xc1\x9e\x14\xc2\xde\xf9\ +\x58\x45\xc7\x7f\x2a\xdc\x15\x6c\x1b\xd8\x16\xb3\xef\xf3\xbe\xeb\ +\x7d\xd9\x39\xb8\x73\xc4\xcc\x5e\x92\xba\x84\xfc\xb8\x7c\x3e\x2b\ +\x17\x59\x57\x6b\xd2\xe6\xd1\x5c\x24\x2d\x0e\xce\x1f\x67\x39\xac\ +\x3b\xc1\x31\x03\xc4\x08\x5e\x44\xc3\x5f\xf4\xb8\x36\x04\x6c\x29\ +\xba\x03\x65\xda\x8c\x63\xe7\x22\x8d\x8d\xc2\xda\xb5\xd1\x4e\x46\ +\xb9\xb9\x70\xdd\x75\x9f\x79\x00\xd5\xa9\xe3\x24\xa0\x87\x06\x16\ +\x17\xeb\x78\x8e\x0f\xd8\x04\xb4\x6b\x23\xc7\x3a\xd4\xc4\xd5\xb6\ +\x67\x28\xb6\x35\x61\x55\x26\x41\xd3\x82\xbb\xef\x02\xc2\x81\x22\ +\xc0\x82\xcf\xf4\xf1\x4c\xe7\x33\x6c\xec\xdb\x78\xc2\xb1\xf8\x76\ +\x7f\xbb\xfc\xa5\xfd\x2f\x34\xfb\x9b\x11\x84\x64\x6b\x32\x17\x66\ +\x5c\x48\x49\xe2\x67\x5b\xc4\x75\xb5\x46\xf9\xa6\x76\xc1\x13\x8a\ +\x72\x91\x59\x59\x27\xb8\xfe\x4d\xcf\x96\xd3\xf9\x7e\xa4\x78\x4d\ +\xe9\x11\x71\xd9\x2b\x8f\xef\x58\xcf\x3d\x17\x1d\x86\x34\x3c\x3a\ +\x66\xce\x9c\x98\x6f\xe7\x98\x80\x44\x5d\x75\x95\xe2\xdc\x73\xa3\ +\x96\x4e\x0d\xb0\x0b\x70\x6b\xdd\x64\x7e\xe8\x55\xce\xb3\xbf\x4b\ +\xaa\x45\xbb\x8f\x43\xde\x79\x18\x9e\x65\x88\xa1\xdf\xbf\x75\x70\ +\x2b\x7f\xed\xf8\x2b\x95\xee\xca\xe3\x06\x8a\x37\xec\xbd\xef\xaf\ +\xed\x7f\xe5\xdd\xbe\x77\xf1\x18\x1e\x14\x8a\xe5\x69\xcb\x59\x9c\ +\xba\xf8\x33\x7d\x7e\x7f\xaf\x21\x7f\xde\x67\xd0\xe2\x96\x11\xdf\ +\xc8\xf5\x65\x16\xa6\x64\x1c\x7f\x6d\x90\xf4\xef\x17\xea\xff\x18\ +\x89\x65\x89\x9e\x32\x3a\xf1\x8b\xc7\xc7\x45\xb6\x6c\xd1\x5c\x64\ +\x78\xe8\x43\x71\x31\xdc\x78\x63\xcc\xa7\xb6\x8f\x1d\x27\x01\xf8\ +\xf2\x97\x61\xfa\x74\x6d\xb3\x87\x81\x0f\x80\x83\x40\x10\x1c\xe6\ +\x20\x37\xdb\xff\xc4\x2c\xeb\x1e\xec\x04\x41\xe2\x91\xfe\x9b\x91\ +\xa1\xd9\x20\x0e\xc2\x12\x66\x6d\xe7\x5a\xfe\xd8\xf6\x47\x2a\x06\ +\x8f\xbd\xcf\x69\xeb\x50\xab\x3c\xd1\xf6\xc4\x3d\x8f\xb7\x3f\x3e\ +\x92\x9f\x52\x14\x57\xc4\x97\xf3\xbf\xcc\xb4\xa4\x4f\xef\x24\xd4\ +\xe5\x33\x5d\x8f\x56\x1a\xec\xe8\xd2\xf9\x2f\x76\x0b\x5c\x59\x6c\ +\x61\x49\xde\xf1\x2f\x97\x84\xfd\x77\x51\xf7\x28\xf4\xed\xd0\xc1\ +\x3c\x8b\x5d\x4f\x04\xcb\x5e\x71\xec\xc7\x72\xb9\xb6\xf3\xd8\x63\ +\xb0\x7f\x36\x65\x73\x8f\x00\x00\x18\x18\x49\x44\x41\x54\xbf\x0e\ +\x83\x24\x24\xc0\x2d\xb7\xc0\xec\xd9\x63\xb2\x95\x63\x06\x12\xb5\ +\x6c\x99\xe2\xfa\xeb\xa3\x0d\xf1\xbb\x81\x0d\x40\x9b\x16\x3b\x33\ +\xd4\x4e\x56\x3b\x9e\x66\x9c\xb5\x05\x0b\x26\xf8\x67\x42\xdf\x2d\ +\x48\x60\x12\x88\x0d\xaf\xe1\xe5\xf1\xb6\xc7\x79\xa0\xe9\x01\x5e\ +\xeb\x7e\x4d\xda\xfc\x9f\x5e\x44\x6d\x98\x46\xc9\x8e\x81\x1d\xb2\ +\xa6\x75\x0d\xbf\x6e\xfa\x35\x0d\x43\x0d\x98\x98\x64\xd8\x33\xb8\ +\x2d\xff\x36\x96\xa7\x2e\xe7\xd3\x8f\x21\x25\x4f\x1f\x30\xd2\x5f\ +\xac\xd3\x75\x36\x8a\x51\x15\x86\x69\x27\x60\xf6\x36\xfc\x75\x0d\ +\x2d\x7f\xd3\x49\x58\xa0\xfb\xc7\x4d\xbc\xed\x63\x7b\xb6\x7c\x22\ +\x3d\xfb\xec\x02\xde\x7c\x53\xd7\x3a\x29\xa5\x03\xac\xd7\x5d\xc7\ +\xe7\x6e\xc2\x38\x00\x5f\xfc\xa2\xce\x6f\x78\xe9\x25\x9d\x73\xb9\ +\x1b\x28\x04\x52\xc0\x92\x6e\xf2\x05\xc7\x5a\x0e\x84\xa7\xf0\xa4\ +\xf9\x25\x7a\x25\x03\x19\xbc\x1a\x6c\xbd\x90\xf1\x27\x94\xa3\x1e\ +\x9f\xe9\xe3\xb9\xae\xe7\xd8\xe3\xd9\xc3\x95\x59\x57\xf2\x5a\xf7\ +\x6b\x52\x14\x5f\x44\xaa\x2d\x95\x78\x4b\x7c\x9f\x4d\xd9\xea\x87\ +\xcc\xa1\x05\x1e\xc3\x43\x4f\xb0\x87\xdf\xb7\xfc\x9e\x97\xbb\x5f\ +\x66\x97\x67\x17\x9e\xb0\x07\x41\xc8\xb4\x67\x72\x43\xce\x0d\xdc\ +\x92\x7f\x0b\xf9\xf1\x9f\xbc\x88\x41\x43\x2e\xfa\xd3\x3e\xa3\xfc\ +\x91\x4a\x93\x16\x8f\xf6\x03\xe7\x3b\xe1\xab\xb3\xac\x9c\x53\x64\ +\x39\xee\x44\x62\x69\x7b\x53\xd8\xfd\xbf\xc0\x1b\x51\x56\xe3\x72\ +\xa0\xf4\x4e\xc8\x5e\x7e\xcc\x1e\x51\x59\xbf\x5e\xf8\xde\xf7\x74\ +\xb4\x5d\x44\x67\xc2\xdf\x71\x07\x6a\xe1\xc2\x31\x2b\x91\x1d\x53\ +\x90\xa8\xa2\x22\x25\x6f\xbe\x29\x1c\x3a\x04\x3b\x77\xea\x6c\xa9\ +\xf5\x40\x36\xb0\x18\x52\x9d\x83\xdc\x19\xbf\x86\x4e\x33\x87\xd7\ +\x42\x57\xe0\x91\x24\xa4\xef\x36\x10\x2b\xa4\xad\x45\xc5\xef\x27\ +\x6c\xf1\x53\xed\xad\xa6\xd6\x57\xcb\xd3\x71\x4f\x33\xdd\x39\x9d\ +\xe2\x84\x62\xd2\x6d\xe9\xe9\x16\x65\x59\xe0\x35\xbc\xb4\xf8\x5b\ +\xa8\xf2\x56\xd1\xe4\x6f\xc2\x6f\xfa\x31\xc4\xc0\x82\x85\xa2\xf8\ +\x22\xae\xce\xba\x9a\xbb\x8a\xee\xa2\x34\xf1\x93\xc7\xc3\x76\x78\ +\x4c\x79\xa2\xca\xe0\xc1\x0a\x83\xba\x7e\x9d\xc6\x90\x1e\x07\x5f\ +\x9a\x6a\xe1\xb2\x89\x16\x8e\xb7\x57\x8a\x74\x7f\x28\xec\xf9\xa1\ +\x0e\x72\x4a\x58\xa7\x26\x8e\x5f\x0d\x85\x57\x70\xac\x73\x7d\xa4\ +\xa6\x46\xf8\xc1\x0f\xa0\xb2\x52\x8b\x99\xc4\x44\x6d\x24\x9c\x7b\ +\xee\x98\xde\xeb\x63\xcb\x49\x00\x75\xc9\x25\x4a\xfe\xf3\x3f\x85\ +\xce\x4e\x1d\x5b\xe8\x37\xe1\x2d\x20\x15\xd4\x0c\xa1\xcc\x51\xc7\ +\x37\xe2\x7f\x87\x47\x92\x78\x3f\x7c\x1e\x3e\x33\x09\xe9\xbb\x1d\ +\xc2\x05\xa8\xb4\x67\xb1\x26\xee\xc4\xb4\x75\x13\x92\x10\x2d\xfe\ +\x16\xda\x02\x6d\xa8\xc8\xbf\x91\xc5\x43\x30\xc5\xc4\xc4\x44\xa1\ +\x48\xb7\xa5\x33\x29\x71\x12\x37\xe6\xde\xc8\x75\x39\xd7\x7d\x6a\ +\xe9\x67\x8d\xcb\x90\xc7\xf7\x19\x3c\x5e\xa5\x5b\x61\x98\x02\xa9\ +\x0e\xb8\xb6\xd4\xc2\x3f\x4c\xb7\x72\xbc\x25\x9d\xd2\xbb\x5d\x8f\ +\x8f\xeb\xd9\xa4\x33\xcf\x2c\x76\xc8\x39\x1f\xca\xbe\x7a\xcc\x8e\ +\x33\xe9\xe8\x10\x7e\xf5\x2b\x1d\xe9\xf5\x78\xb4\x35\xb3\x62\x05\ +\xdc\x76\x1b\x6a\xc2\x04\xf5\xb9\x06\x09\x00\xb7\xde\xda\x47\x5d\ +\x5d\x3a\x4f\x3e\xa9\xb3\xa6\x1a\xd1\x40\x71\x82\xa5\x54\x58\x64\ +\xdf\xc6\x3f\xf3\x00\x32\x04\x1f\x84\x57\x69\x8e\xe2\xbe\x1c\x09\ +\x4c\xc7\x9e\xfe\x1a\x71\x29\x1b\x88\x4b\x68\x20\xa0\x7a\xf0\x19\ +\x5e\xc2\x12\x3e\xac\x82\xcf\xa2\x2c\x24\x58\x13\x48\xb7\xa7\x53\ +\xe0\x28\x60\x51\xca\x22\xae\xcf\xb9\x9e\x65\xe9\xcb\xd4\x37\xf9\ +\xe6\xc7\x7e\xad\xc1\x80\xf9\xcc\x8e\x4e\x59\xfd\xf3\x6d\x06\xaf\ +\x37\xea\xc6\x7a\x22\x90\x1e\x0f\x57\x4c\xb4\xf0\xad\x79\x56\x26\ +\x1f\xa7\x35\x23\x3d\xdb\x84\x7d\x3f\xd7\x8d\xf3\xc2\xee\xc8\x74\ +\xb0\x45\x30\xf5\x3b\xa8\x8c\x63\x9b\xd9\x2b\x86\x51\xc2\x83\x0f\ +\xc2\xda\xb5\xd1\xf8\xcc\xe4\xc9\xf0\x8d\x6f\x8c\xa9\x98\x39\xa9\ +\x20\x51\x39\x39\x19\x52\x59\x29\xb4\xb7\xc3\x1b\x6f\x68\xfd\x64\ +\x2f\x90\x04\x38\xc0\x3a\xce\xe4\x1c\xfb\x07\xd8\x08\xe1\xf4\xfb\ +\x78\x37\x74\x01\x7d\x92\x8e\x19\x28\xc1\xdf\x75\x17\x59\xa1\xab\ +\x28\xca\xd9\x8e\xdd\xb9\x1b\x49\x68\x20\x6c\xe9\x27\x2c\xba\xaa\ +\xcf\xa6\x6c\x24\x59\x93\x28\x8a\x2b\x62\x46\xd2\x0c\x96\xa5\x2e\ +\x63\x7e\xea\x7c\xf5\x73\x7e\xfe\x89\xe0\x38\xe0\x92\xd5\x6b\xf6\ +\x18\x3c\x7b\xd0\xa4\xba\x4f\x46\x72\x56\xf3\x9d\x70\x75\x89\x85\ +\xaf\xcf\xb1\x32\xed\x38\x3b\x2f\x4a\xe7\x7a\xa1\xf2\xc7\xa3\x26\ +\x94\x5a\x23\xd3\xc1\xbe\x87\xca\xbf\xf0\xd8\x8f\xf9\x97\xbf\xd4\ +\xb1\x66\x8d\xe6\xc4\xc3\x7a\xc8\xd7\xbe\x06\x97\x5c\x72\x52\x1a\ +\xfe\x9c\xd4\x7e\x20\x52\x5e\x2e\xfc\xe4\x27\x7a\xc0\xb1\xdf\x0f\ +\x76\x60\x25\x70\x89\x56\x68\x0d\x8b\x85\x03\xc6\x34\xfe\xe0\xbf\ +\x83\x37\x82\x97\xd3\x62\x16\x11\xc2\x8e\x52\x82\xd3\x6e\xb0\x28\ +\x57\x58\x5e\xe4\x65\x5c\x7a\x3f\x29\xf1\x5e\xd2\xe2\x4d\xb2\xe2\ +\x12\xc9\xb2\xa7\x53\x90\x50\xf0\xa9\x4a\x69\xe3\xa0\x94\xb7\xb8\ +\x85\xad\x9d\x26\xaf\x1f\x32\xa9\xec\x8d\xf6\xa9\x77\x58\x75\x6d\ +\xcd\x4d\x93\x2c\x7c\x79\x9a\x85\x09\xc7\xd1\x2b\x45\xc2\xfe\xbb\ +\x68\x7f\x7d\x0d\x07\x7e\x03\xbd\x5b\x74\xf0\x4e\xd9\xf4\x9c\x9b\ +\xe9\xf7\xa2\x26\xde\x72\xec\xc7\x7c\xe6\x19\xe1\xa7\x3f\x8d\xea\ +\x21\x19\x19\x70\xc7\x1d\xf0\xad\x6f\x8d\xb9\x98\x39\x25\x20\x01\ +\x90\xa7\x9f\x16\x7e\xf9\x4b\xd8\xbd\x5b\x97\x60\x38\x80\x73\x81\ +\x0b\x80\x02\x30\x6d\x16\xba\xcd\x2c\xd6\x06\x6e\xe2\x85\xe0\xf5\ +\x54\x1a\xb3\x18\x34\x93\x31\xd1\xd9\xf5\x71\x36\x28\x4b\x13\x16\ +\xe4\x28\x66\x65\x29\x26\xa6\x58\xc8\x88\x57\x24\xda\x15\x0e\x6b\ +\x34\x03\x3f\x6c\x42\xd0\x00\x6f\x48\xe8\xf5\x0b\xf5\x03\x42\x45\ +\x97\xb0\xad\x53\x68\xf5\x68\xff\xc7\x70\x2b\xf2\xec\x04\x98\x93\ +\xad\xb8\x6d\x9a\x95\xab\x4b\x8f\xaf\x6d\xb8\xb8\xeb\x85\xd6\x57\ +\xa0\xee\x11\x18\x3c\xa0\x8b\xbe\x2d\x0e\x48\x9b\xa3\xc7\xc7\x1d\ +\x0f\x40\x9e\x7b\x4e\xf8\xd9\xcf\xf4\x5a\x85\x42\xba\xcf\xc8\xea\ +\xd5\xf0\xbd\xef\x11\xab\xc1\xd0\xa7\x25\x48\x00\xe4\x91\x47\x84\ +\x07\x1f\xd4\xce\xa0\x50\x48\x03\x65\x65\x04\x28\x45\x20\x76\x45\ +\x50\x1c\xec\x30\x97\xf0\x5c\xe8\x8b\x6c\x0e\x9f\x43\x7d\x78\x3c\ +\x6e\xc3\x31\xb2\xb1\x16\x05\x36\xa5\x27\x4e\xe4\x25\x2a\x72\x9c\ +\x90\xea\x88\xf6\x96\x1f\x0a\xeb\x3c\xda\x0e\xaf\xd0\xe5\xd3\x1c\ +\x23\x2c\xd1\xb1\x27\x16\x05\x19\xf1\xba\x7c\xf3\xa2\xf1\x16\x6e\ +\x98\x64\x39\xee\x01\x04\xd2\xb5\x51\x68\xf8\x8b\x1e\x24\xe0\xef\ +\xd0\x49\x44\x36\x67\x54\x07\x19\x77\xed\xb1\xeb\x20\x6b\xd7\xd6\ +\xf1\x8b\x5f\xc0\x9e\x3d\x7a\x8d\x9c\x4e\x9d\x02\xf0\x6f\xff\x86\ +\x9a\x37\xef\xe4\x8e\x60\xe1\x14\x91\x3c\xf4\x90\xf0\xfb\xdf\xeb\ +\x6a\xb3\x50\x48\x8b\x9e\xc5\x0a\x2e\x54\x50\x2c\xe0\x10\x4c\xac\ +\x78\x2c\x99\x7c\x68\xb9\x8a\x57\xc3\xd7\x52\x11\x9a\x43\x4b\x38\ +\x9b\x5e\xbf\x95\x80\xa1\x03\xcb\xc3\x33\x6d\x54\xe4\x6a\xd4\x88\ +\xc5\xa3\xff\x13\xa2\xb3\x70\x14\x90\x68\x87\xac\x04\xc5\xb8\x24\ +\x58\x56\x60\xe1\xaa\x12\x0b\x4b\xf2\x8f\x13\x1c\xfd\xfb\x84\xae\ +\xf5\x70\xe8\x09\xe8\xdf\x1d\x29\x42\x13\x3d\xbd\x33\xe7\x5c\x98\ +\xf2\xcf\xa8\xdc\x55\xc7\x06\x90\xae\x2e\x17\x2f\xbe\x98\xce\x03\ +\x0f\x44\x6f\xa2\xc8\x9c\x65\xbe\xff\x7d\xd4\xe2\xc5\x27\x7d\xcf\ +\x4e\x69\x8f\x32\x79\xe8\x21\xe1\xe1\x87\xa3\x8b\x61\x03\x66\xd9\ +\xe0\x22\x0b\x4c\x31\x21\x41\xa7\x82\x19\xd8\x18\x4a\x28\xa3\x2a\ +\xf1\x7a\x36\x9b\xe7\xb3\xd3\x3f\x85\xa6\x60\x16\xae\x80\x95\xc1\ +\x20\xf8\xc2\x5a\x7c\x18\xa3\x8a\xbf\x2c\x4a\x67\xe6\xc7\x59\xc1\ +\x69\xd7\x23\x4f\x32\xe2\xa0\x34\x4d\xb1\x30\xd7\xc2\x8a\x02\xc5\ +\xcc\xe3\xec\x53\x2f\x03\xd5\x42\xdf\x2e\x68\x7e\x0e\x3a\xdf\x43\ +\x97\x44\x18\x5a\xff\x48\x2a\xd6\xd3\x3b\x4b\xef\x40\xa5\x1d\x5b\ +\x23\x1d\xa9\xaf\xd7\xf1\x98\x47\x1e\x81\x43\x87\x74\x1a\x68\x72\ +\xb2\x06\xc8\x3d\xf7\xe8\x69\x9d\xa7\x80\x4e\x79\x23\x3b\x59\xb3\ +\x46\x78\xe4\x11\x3d\x2b\x2e\x10\xd0\x81\x82\xd2\x04\xb8\x24\x0e\ +\x66\x03\x4e\x2f\xa8\x10\x60\xc1\x54\x76\x8c\x94\xe9\x78\x73\xae\ +\xa6\x21\x6e\x05\x07\x42\xc5\x1c\x0a\xe4\xd1\xe6\x8f\xa3\x2f\xa0\ +\xf0\x85\x74\x6b\x08\x85\x56\x44\x9d\x76\x5d\x98\x5e\x98\xa4\x28\ +\x4b\x53\x4c\xcd\x50\xc7\x6f\xb1\x84\xbc\xf7\xe1\xa9\xbb\x87\xfe\ +\x4a\x68\x7b\x45\x0f\xbe\x0e\x74\x83\x19\xd6\xdc\xc3\x91\xa1\x2d\ +\x98\xe2\xdb\xb5\xa3\x2c\xe1\xd8\xca\x53\x65\xd7\x2e\xe1\xb1\xc7\ +\xe0\xd9\x67\xf5\x30\x23\xd3\xd4\xb5\x4c\x97\x5e\x0a\xff\xfa\xaf\ +\x27\xc5\xd4\x3d\x6d\x41\x02\x20\xb5\xb5\xc2\x9f\xfe\x04\x8f\x3f\ +\x1e\xad\x1b\x89\x77\xc0\xc4\x64\xb8\xac\x18\x16\x04\xc1\x7e\x08\ +\x4c\x6f\xa4\x88\x49\xe9\x56\xdb\x09\xf9\xe0\x2c\x46\xd2\x66\x23\ +\x19\x4b\x30\xd3\xe6\x62\x4f\x2b\x8b\xc9\x35\x89\x11\xbc\x88\xbe\ +\x8a\x72\x7a\xb7\x81\x6b\x1b\x0c\x56\xeb\x51\xb5\x81\xee\x48\x27\ +\x22\x05\x16\xab\x9e\xbe\x51\x74\x9d\x1e\xd3\x92\xbd\xf2\xee\x63\ +\x9d\xa5\x27\x9b\x37\x0b\x8f\x3e\x0a\xe5\xe5\x7a\x2e\x4d\x38\xac\ +\x83\xa2\xab\x56\xc1\x37\xbf\x09\xd7\x5d\x57\x7a\x22\x6d\x23\xce\ +\x18\x90\x00\x48\x6b\xab\xf0\xea\xab\xf0\xdb\xdf\xea\x7c\xcd\x60\ +\x50\xe7\xa3\xe4\x67\xc0\x82\x22\xb8\xb2\x0c\x72\x9a\x20\x78\x40\ +\x37\x9e\x13\x53\xdf\xc1\xca\x06\x8e\x54\x1d\x0f\x89\xcf\xd5\x75\ +\x2b\x29\x93\xc1\x59\x02\xce\x71\xc3\xaf\x5d\xfc\x49\x2e\x70\xf1\ +\xb5\x0b\x81\x1e\x9d\xb9\xee\x6d\xd4\x0d\x77\xdd\xb5\xfa\x77\x7f\ +\xa7\x06\xc6\xb0\xbe\xa1\x2c\xda\xef\x11\x5f\x00\xf9\x97\x40\xe1\ +\x15\x90\x3e\xff\x98\xa7\x92\x4a\x47\x87\xf0\xec\xb3\x3a\x27\x64\ +\xe7\xce\x68\x29\x44\x62\x22\xdc\x70\x03\x7c\xe5\x2b\x3a\x1d\xf4\ +\x34\xa0\xd3\x6a\xaa\xb7\x18\x46\x09\xaf\xbd\xa6\x1d\x47\xeb\xd7\ +\xeb\x85\x13\x81\xe4\x24\x28\xc9\x87\x0b\x66\xc1\x45\x25\xe0\xa8\ +\x86\x81\x6d\x10\x8a\xe8\x02\xc3\x80\x41\x81\x2d\x01\xec\xa9\x3a\ +\x6f\xd4\x96\x04\xb6\xc4\x48\x3b\x87\xc8\xb3\xc5\xa1\xdf\x27\x46\ +\xa4\xd3\xa1\x4f\xbb\xcc\xc3\x3e\xed\xf8\x0a\x7b\x20\x38\x00\xe1\ +\xc1\xa8\x28\x41\x69\x60\x58\xec\x90\x3c\x09\xf2\x2e\x82\x9c\xf3\ +\x20\x6d\x06\x2a\xb9\xf4\xd8\x4d\xdb\x57\x5f\x15\x9e\x7e\x1a\x36\ +\x6c\xd0\x81\xba\x50\x48\xbb\xd9\x0b\x0b\xe1\xf6\xdb\x61\xf5\x6a\ +\xd4\x9c\x39\xa7\xcd\xde\x9c\x96\xa3\xdf\x65\xd7\x2e\xe1\xa9\xa7\ +\xe0\xa9\xa7\xa2\x2c\xd8\x62\xd1\xe9\x79\x33\x4a\xe0\xe2\x85\x70\ +\xce\x64\xb0\x37\x42\xff\x56\xe8\xaf\xd4\x9b\x2a\x26\x60\x1e\xa5\ +\xe7\xd8\xf0\x26\xdb\xa2\x3d\xdc\xc4\x8c\x00\x6c\x34\xc8\x46\xbf\ +\xdf\x12\x79\xaf\x45\x73\xa4\x8c\x45\x90\x73\x0e\xa4\xce\x84\xa4\ +\x62\x94\x73\xdc\xb1\x83\x63\xd3\x26\xe1\xf9\xe7\x75\xfc\xa5\xba\ +\x3a\xda\x8d\xc8\x6e\x87\xe5\xcb\xe1\x2b\x5f\x81\x0b\x2f\x8c\x69\ +\x12\xf3\x19\x0b\x12\x00\x69\x6e\x16\x36\x6d\x82\xff\xfe\x6f\xd8\ +\xbc\x59\x17\x41\x9b\xa6\x5e\xd0\xa2\x22\x98\x31\x15\x2e\x58\x0c\ +\xab\x66\x43\xba\x80\xaf\x06\x06\xf7\xc1\xc0\x3e\xdd\x7a\xdb\xf0\ +\x45\x37\x5f\x64\x14\x08\x8e\x36\xff\x53\x45\xda\x9f\x47\xc0\x14\ +\x97\xa5\x0b\xb7\x53\xa7\xeb\xe1\x09\x49\xa5\x90\x90\x8f\x4a\x3d\ +\x3e\x07\x96\xac\x5f\x2f\xbc\xfc\x32\x6c\xdc\x08\x07\x0e\x40\x5f\ +\x9f\xbe\x16\xab\x15\x0a\x0a\x74\x24\xf7\xba\xeb\x50\xe7\x9c\x73\ +\x5a\xee\xc7\x69\x0b\x92\x91\x05\xde\xb9\x53\x78\xe5\x15\x78\xfa\ +\x69\xdd\x93\x2d\x10\xd0\x9b\xee\x70\x68\xf6\x3c\x79\x32\xac\x5a\ +\x09\xab\x16\xc2\xc4\x2c\xb0\x05\x20\x3c\x00\x81\x76\xdd\xbe\x7c\ +\xa8\x53\x9b\xa8\x21\xb7\x06\x8e\x19\x8a\xe8\x16\x56\x5d\xa0\x6d\ +\x73\x6a\xf1\x14\x97\x15\x51\x84\xc7\x69\x3f\x87\x23\x1d\x1c\x19\ +\xc7\x3d\x01\x5d\xda\xdb\x85\x2d\x5b\x74\x65\xdd\x8e\x1d\xfa\xbb\ +\xf7\xf7\x6b\xa5\xdc\x6a\xd5\xba\xc7\x05\x17\xc0\xcd\x37\xc3\x8a\ +\x15\x27\xcd\xc5\x7e\x46\x82\x64\x44\xc9\xab\xac\x84\x17\x5e\x80\ +\x97\x5f\xd6\x22\x28\x14\xd2\x60\xb1\xd9\x74\xad\x49\x51\x11\xcc\ +\x9d\xab\xd9\xf6\x92\xc5\x90\x97\x01\xb6\xb0\x4e\x15\x94\x60\xa4\ +\xc3\x61\xf8\x70\xfd\x65\x58\xcf\xb0\x38\x74\x33\xbb\xf8\xec\xbb\ +\xd5\x09\x14\xab\x8b\xdf\x7f\x17\x5b\xb7\xae\x61\xf3\x66\xcd\xfd\ +\x0e\x1e\xd4\xd6\xda\x30\x17\xb4\x58\x74\xab\xcc\x85\x0b\xe1\xc6\ +\x1b\x61\xd5\x2a\x5d\x6c\x7f\x9a\xd3\xe7\x02\x24\x23\x9b\xd0\xd0\ +\x20\xec\xdd\x0b\xaf\xbc\xa2\xef\xd0\x61\xb0\x0c\x37\xda\x4f\x49\ +\xd1\x80\x19\xe6\x30\xf3\xe6\xc1\xac\x59\x50\x56\x36\x66\xa9\x7d\ +\x52\x53\x23\x54\x57\xeb\xac\xf5\x3d\x7b\x74\x7b\xcc\xf6\x76\x1d\ +\xd2\x0f\x46\x86\x09\x0e\x83\x63\xde\x3c\xb8\xf6\x5a\x58\xb9\x12\ +\x16\x2c\xb8\x38\x56\x65\x98\x67\x41\xf2\x71\x9e\xc9\xea\x6a\x5d\ +\xde\xf8\xe6\x9b\x50\x5f\xaf\x8b\x93\x4c\x53\x73\x17\xa5\x20\x29\ +\x49\x77\xfa\xc9\xcc\x84\xec\x6c\x5d\xf6\x58\x52\xa2\x9f\xc7\x8d\ +\xd3\xb9\xb7\xf9\xf9\x77\xab\xf8\xcf\xc6\x39\xa4\xa3\x43\x70\xb9\ +\x74\x8d\x4b\x53\x93\x06\xc3\xc1\x83\xfa\xf7\xde\x5e\xdd\x0d\xbb\ +\xaf\x4f\x03\x43\x44\x03\xc3\x6a\xd5\xc0\x5d\xb4\x08\xae\xb8\x42\ +\x3f\xcf\x9e\x7d\xbf\x72\x3a\xef\xfd\x3c\xad\xf7\xe7\x12\x24\x87\ +\x29\xb7\x0d\x0d\x5a\xe6\xaf\x5f\xaf\xfb\xb5\x75\x75\x45\xb9\xcb\ +\xe8\xa9\x5e\x4e\xa7\x76\x71\x27\x27\x6b\x00\x25\x26\xea\x0d\x4c\ +\x4a\xd2\x7f\x4b\x48\x88\x8e\xfc\x10\xd1\x16\x55\x20\xa0\xc1\xe7\ +\xf5\xea\xa4\x63\xaf\x57\xcf\xb3\x73\xbb\xf5\x63\x70\x30\xaa\x23\ +\x81\x06\xc6\x30\xd7\x28\x2b\xd3\x99\x63\x2b\x57\xea\x8a\xba\x59\ +\xb3\x4e\xb9\x53\xec\xef\x12\x24\x23\x60\x09\x06\x2f\xe2\xe0\xc1\ +\x72\x9a\x9a\xa0\xa2\x02\xb6\x6d\xd3\xe1\xf5\x8e\x8e\xe8\x9c\x40\ +\x91\xe8\x63\x98\xac\x56\xad\xd3\x0c\x3f\x2c\x96\xe8\x90\x27\xd3\ +\x8c\xce\xae\x0b\x87\xa3\x3a\xd0\xc8\xca\xa9\xa8\x28\xb1\x58\x34\ +\xd8\x4a\x4b\xb5\x48\x59\xb2\x44\x37\x1c\x2c\x2a\x42\x4d\x9c\xf8\ +\xb9\x5f\xe3\x33\x02\x24\x87\x01\x66\x70\xf0\x19\x9a\x9b\x57\xd3\ +\xd1\xa1\x83\x64\x35\x35\x3a\x80\x58\x5b\xab\xb9\x8c\xd7\x7b\xf8\ +\xec\xc0\x23\x81\x33\xfc\xf3\x30\x58\x46\x3f\x0f\x3f\x1c\x0e\xcd\ +\x85\x8a\x8a\x34\x18\xa6\x4e\xd5\x3a\x50\x51\x11\xe4\xe6\xc6\xbc\ +\xcc\xf2\x2c\x48\xc6\x1a\x34\x2d\x2d\x42\x5f\x9f\x16\x17\xbd\xbd\ +\x5a\x87\xe8\xe8\xd0\x41\xb4\x9e\x1e\x2d\x32\xdc\x6e\x2d\x56\x42\ +\x21\xcd\x35\x94\xd2\xa2\x27\x2e\x4e\x8b\xa2\xa4\x24\x1d\x6c\xcb\ +\xce\xd6\x0e\xbd\xc2\x42\xfd\x9c\x96\xa6\x1f\x79\x79\x6b\x55\x4a\ +\xca\x4d\x67\xea\x1a\x9e\xf1\x20\xf9\x58\x25\xd4\xe7\xd3\x29\x94\ +\xc1\xa0\x06\xc7\xb0\x68\x19\xad\x5f\x58\xad\x1a\x2c\x76\xbb\xe6\ +\x1e\x09\x09\x90\x99\xf9\xb9\x53\x3c\xcf\xd2\x59\x3a\x4b\x67\xe9\ +\x2c\x9d\xa5\xb3\x74\x96\xce\xd2\x49\xa0\xd6\x0e\x8f\xec\xaf\xed\ +\x95\xaa\x9a\x1e\xf1\xfa\x42\xf7\x9d\xce\xdf\xd5\x1f\x08\xdf\x55\ +\x5d\xef\x92\xaa\x9a\x1e\x69\xed\x70\xcb\x99\xb2\x07\xb6\x93\x79\ +\xb2\x0d\x5b\x5b\xe4\xf5\xf7\x1b\xe9\x1b\x08\x50\x98\x97\xc4\xed\ +\x37\x4c\x63\x5c\x41\xf2\x27\x5a\x58\x7f\x59\x77\x80\x77\x36\x35\ +\x33\xe4\x0f\xf3\x8d\xdb\xe7\xdc\x03\x1c\x97\x65\xf1\xd4\x4b\xd5\ +\x52\x53\xdf\x47\xd8\x18\x5d\x1e\x0a\x89\x09\x76\x0a\xf3\x92\x58\ +\x38\x3b\x87\x29\x25\x19\x27\x64\xed\xbd\xfa\xee\xa1\x35\x7f\x7d\ +\xa1\x9a\xee\xde\x21\x2e\x3f\x6f\x22\xed\x5d\x5e\xc9\xcf\x71\xc6\ +\xd4\x82\xac\x6f\xea\x17\xaf\x2f\x84\x52\x8a\x99\x53\xb2\xd4\x19\ +\x07\x92\xfa\xa6\x01\x5e\x7a\xfb\x10\xad\xed\x1e\x66\x4e\xcd\xe4\ +\x9a\x8b\x4a\x3e\xd3\x67\xb6\xed\xee\xc4\xe3\x0b\x71\xfd\x65\x65\ +\x18\x86\x59\x62\xb5\x5a\x8e\xd9\xbd\xfd\xce\xc6\x66\xca\x3f\x68\ +\x22\x18\x32\x23\x3e\x31\x5d\x72\x6e\xb1\x28\x9c\x89\x76\xc6\x17\ +\x24\xf1\xd2\x5b\xf5\x72\xf5\x45\xc7\xef\x08\xeb\xec\xf1\xb1\xb3\ +\xb2\x9b\xd6\x0e\x0f\x53\x4a\xd2\x09\x06\x8d\x98\xaf\xe1\x6f\x1e\ +\xab\xa0\x62\x5f\x37\x0e\xbb\x95\x4d\x3b\xda\x64\xf9\x82\x02\x75\ +\x46\x81\x24\x14\x16\x7c\x43\x61\x3c\xbe\x10\x43\x43\x61\x4c\xf3\ +\xe4\x71\xe4\x41\x6f\x90\xde\x3e\x3f\x81\xa0\x41\x51\x7e\x12\x05\ +\x39\x49\x84\xc2\x06\xf5\x4d\x03\x74\x74\x7b\x69\x68\x1e\x20\x3e\ +\xce\x46\x75\x9d\x4b\xa6\x94\x66\xc4\x26\x99\x5a\x62\x7f\x7d\xfb\ +\x6a\x5d\x6c\xa9\xe8\x20\xde\x61\xa3\xb7\xdf\x7f\xe6\x89\x9b\x4f\ +\xa3\x0f\xb6\xb5\xca\xd6\x8a\x4e\x06\xdc\x01\x26\x14\x26\x73\xde\ +\xf2\x22\x7e\xfa\xd0\xb6\x4f\xd4\x01\xde\xdf\xd2\xb2\xa6\xaa\xba\ +\x17\xd7\x40\x80\x38\x87\x95\xd2\x09\xa9\xac\x58\x58\xc0\x84\xc2\ +\x94\x8f\xdd\xe8\xf3\x97\x15\x71\xcb\xd5\x53\x48\x49\xb2\xb3\xee\ +\xcd\x3a\xfe\xf4\xb7\x03\x74\xf5\x0c\xb1\x7d\x6f\x27\xae\x81\x8f\ +\x2e\xfc\xa1\xe6\x01\xd9\xba\xbb\x93\xda\x86\x7e\x7c\x43\x21\x92\ +\x9c\x76\x26\x17\xa7\xb3\x74\x5e\x3e\x85\x79\x47\xef\xd7\x69\x8a\ +\xd0\xd1\xe3\xe3\xc9\x17\x0f\x48\x4d\x7d\x3f\x76\xbb\x85\xc5\x73\ +\xf2\xb8\x68\xe5\x47\x07\x14\x0d\x7a\x02\xcf\x6c\xdc\xd6\xb6\xba\ +\xea\xa0\x8b\xfe\xc1\x00\x0e\xbb\x85\xf1\x85\xc9\x2c\x9b\x9f\x3f\ +\x22\x02\xab\xeb\x5c\xb2\xa3\xb2\x8b\xfb\x7e\xa7\xd7\x23\x6c\x98\ +\xbc\xff\x61\x0b\x8f\x3d\x53\x25\x56\xab\x62\xf9\x82\x7c\x26\x4d\ +\x4c\x57\x67\x34\x48\x9e\x78\x6e\x9f\xfc\x9f\xdf\x6c\xa1\xb6\x61\ +\x80\xb0\x61\x92\x92\xe4\xe0\xfd\x2d\xad\xd4\x35\x0d\x60\x1c\x85\ +\xe3\xec\x3b\xd8\x2b\xf7\xde\xf7\x01\x9b\x76\xb6\xe3\xea\x0f\x10\ +\x0a\x19\x23\xa2\x63\xe6\xe4\x4c\x3e\xd8\xd6\x2a\x2b\x17\x1d\xbd\ +\x2f\x49\x6e\x56\x22\x33\x26\x67\x90\x9f\x93\xa4\x5e\x28\xaf\x93\ +\x17\xca\x0f\xd1\xd5\x33\xf4\xd1\x54\x57\xad\x67\xc8\xbf\xfd\x7c\ +\x23\x55\x35\x2e\xdc\xde\x20\xa6\x21\x58\xad\x8a\x94\x24\x07\xb3\ +\xa6\x66\xb1\x79\x67\xbb\x2c\x9b\xff\xd1\x5c\x95\xe6\x36\x37\x3f\ +\xff\xfd\x76\x1a\x5a\x06\xe9\xed\xd3\xc0\x5b\xf7\x46\x1d\xbf\xff\ +\xf3\x6e\xb9\xed\x86\xe9\xf7\x3b\x13\xed\xf7\x02\xec\xde\xd7\x2d\ +\xff\x76\xff\x46\xb6\x56\x74\xe8\xeb\x08\x6b\x71\xe8\x4c\xb4\x33\ +\xb1\x30\x99\xe7\x5f\x3b\x28\x37\x5c\x3e\x49\x55\xec\xeb\xe6\xd1\ +\xa7\xab\x68\x6a\xf3\xe8\x20\xb5\x61\xf2\xdc\x6b\xb5\xbc\xf6\x5e\ +\x23\x0a\xf8\xbf\xff\xba\xec\xcc\xe6\x24\xdb\x76\x77\xc8\xbf\xff\ +\x62\x13\x1f\xee\xec\xc0\x14\x61\xde\xf4\x6c\x8a\xf2\x93\xa8\xa9\ +\xef\xe3\x50\xcb\x20\x81\x23\x64\xbb\x3f\x10\xbe\xeb\xbb\x3f\xdd\ +\xc0\x73\xaf\xd5\x32\xe8\x09\xb2\x7c\x41\x3e\xf3\x67\xe4\xd0\xd2\ +\xe1\xe6\x9d\x4d\x2d\xbc\xfe\x7e\x23\xa6\x29\x1c\xa8\xed\x95\xa9\ +\x65\x99\x1f\xd9\x40\x8f\x2f\x44\x67\xb7\x8f\x9a\xfa\x3e\xf9\xe3\ +\x73\xfb\x18\x70\x07\xb0\x58\x14\xf3\x67\xe6\x90\x36\xaa\x69\xfc\ +\xce\xca\x4e\xf9\xd1\xaf\xb7\xb0\x7e\x4b\x2b\x86\x61\xb2\x60\x56\ +\x2e\x13\x0a\x93\xa9\x3a\xd8\xcb\xc1\x43\xfd\x34\xb5\xba\x31\x0c\ +\x93\x9a\x7a\x97\x4c\x3e\x42\xe9\xad\xd8\xd7\x43\x41\xae\x93\x49\ +\xc5\x69\xa4\x24\xf9\xa8\xac\xee\x65\xcf\xfe\x1e\x1e\x7a\x62\x0f\ +\x93\x4b\xd2\xef\x01\xee\x75\x0d\xf8\xb7\xff\xfb\xfd\x1b\x59\xf7\ +\x66\x1d\x03\xee\x20\x53\x4b\xd2\x99\x39\x25\x93\xf6\x2e\x2f\x15\ +\xfb\x7b\x68\x6c\x71\xd3\x37\x10\x60\xc3\xb6\x16\xe9\xea\x19\x22\ +\x3d\x35\x0e\xbb\xdd\xc2\x90\x1f\x14\x8a\xd4\xe4\x38\xd2\x52\x1c\ +\x28\xa5\x88\x8f\xb3\x9d\xd9\x20\x79\x7b\x63\x13\x07\x6a\x5d\x04\ +\x43\x06\x2b\x17\x15\xf0\xed\x7f\x9c\x47\xd9\xc4\x34\x2a\xab\x7b\ +\xf8\xf1\x83\x5b\xe9\x1f\x0c\x1c\xf9\xfe\x35\x1b\xb6\xb6\xe2\xea\ +\xf7\x33\x77\x46\x36\x5f\xbf\x75\x36\x73\xa6\x65\xe3\xf6\x06\xf1\ +\xfa\xc2\xbc\xb3\xa9\x85\xb7\x37\x35\x73\xfd\xa5\x47\x6f\x73\xf6\ +\xca\x3b\x0d\x54\xec\xeb\xc1\xa2\xa0\xb1\x65\x10\xb7\x27\xc8\xd2\ +\x79\x79\xdc\xf9\xc5\x19\x4c\x1b\x05\xaa\x57\xdf\x6d\x60\x67\x65\ +\x17\x43\xfe\x30\x17\x2c\x1f\xc7\x3f\x7f\x65\x2e\xa5\xe3\x53\xa9\ +\x6d\xec\xe7\x7f\xff\xe7\x66\xaa\xaa\x5d\x94\x6f\x68\xe2\xfa\x4b\ +\xcb\x3e\x72\x8e\x38\x87\x85\xaf\xdf\x36\x9b\xe5\xf3\xf3\xf1\x0e\ +\x85\xf8\xc9\x83\x5b\x35\x67\x6c\x1c\x60\xfd\x96\x56\xba\x7a\x7d\ +\xae\x0d\x5b\x5a\xd3\x37\xee\x68\xa7\x7f\x30\xc0\xc4\xa2\x14\xfe\ +\xe7\x5d\xf3\x59\x38\x3b\x17\xb7\x27\xc8\x8f\x1f\xdc\xc2\x7b\x1f\ +\xb6\xb2\xe7\x40\x0f\xef\x6c\x6c\xe6\xc6\x2b\x26\x71\xef\xd7\x17\ +\xd2\xda\xe1\x61\x57\x55\x37\x76\xbb\x85\x3b\x6e\x9e\xc1\x92\xb9\ +\x79\xd8\x6c\x8a\x29\x25\x19\x63\xd6\xab\xc4\x72\x3a\x80\xa4\xaa\ +\xc6\xc5\xa0\x47\xcf\xbc\x39\x77\x49\x21\x2b\x16\x16\xec\x98\x56\ +\x96\xa1\x56\x5f\x39\x59\x95\x4e\x48\xc5\x6e\x3b\xfc\x6b\xee\xde\ +\xdf\x43\x6f\x9f\x1f\x11\x18\x1a\x0a\xf3\xd6\x07\x4d\x3c\xfa\x74\ +\x25\x4f\xbf\x54\x43\x47\xb7\x0f\x11\xc1\xe3\x0d\xb1\xaf\xd6\x45\ +\x47\xb7\xf7\x23\xb2\xaa\xad\xd3\xc3\xce\xbd\x9d\x6c\xdd\xdd\x49\ +\x5b\x97\x17\x94\x62\x6a\x69\x3a\xd3\xcb\x32\x3e\x72\x9e\x81\xc8\ +\xd8\xce\xf3\x96\x15\xb2\x6a\x71\xe1\xfd\x93\x4b\xd2\xd5\x15\xe7\ +\x17\xab\x19\x93\x32\x89\x8b\xb3\xe2\xf6\x86\x38\x50\xe7\xc2\x35\ +\xe0\xdf\x3e\xfa\xb3\x33\xa7\x64\xb2\x62\x41\x3e\x33\xa7\x64\xa9\ +\x25\x73\xf3\xd5\x8a\x45\x05\xa4\x26\xc7\x61\x9a\xc2\xbe\x83\x2e\ +\x06\x06\x03\xe9\xbb\xf7\x77\xe3\xea\xd7\xd7\xb1\x60\x56\x0e\x4b\ +\xe7\xe5\x33\xb5\x34\x43\x2d\x9a\x93\xa7\x56\x2c\xd4\xef\x0f\x1b\ +\xc2\x9e\xfd\x3d\xd8\xed\x56\x16\xce\xce\x53\xc9\x49\xba\x6e\xc8\ +\x62\x51\x94\x4e\x48\x65\xf1\xdc\x3c\x35\x7f\x66\xae\x1a\x16\x5f\ +\x67\x2c\x27\x19\xf4\x04\x09\x47\xa6\x52\xe5\x64\x26\x92\x91\x16\ +\x3f\x32\xed\x29\x2d\x39\x0e\x9b\xd5\x42\x70\xd4\x6c\xb3\xfe\xc1\ +\xc0\xc8\xef\xcd\xed\x1e\x5e\x7a\xeb\x10\x96\x48\x5b\xb3\x21\x7f\ +\x38\x92\xf2\x61\x65\xd0\x1d\x24\x14\xfe\xe8\x5c\x9b\x0b\x96\x15\ +\x71\xce\xe2\x42\x9a\xda\xdc\xbc\xf2\xf6\x21\xba\xfb\xfc\xbc\xb3\ +\xa9\x99\x25\x73\xf3\x8e\x54\x28\x09\x1b\xe6\x88\x1e\x33\x7a\x23\ +\x46\xb3\xfe\xfe\xc1\x00\x43\x43\xa1\x05\xa3\x3f\x9b\x9d\x91\x40\ +\x42\x7c\x74\x79\x73\x32\x13\x89\x8b\xf4\xc5\x18\x74\x07\x09\x87\ +\x4d\xfa\x07\xb5\x2e\x35\xfc\xf7\xf8\xb8\xe8\x78\x95\xec\x51\xef\ +\xef\x77\x07\x09\x85\x8e\xb8\x8e\x93\xe8\xaa\x3b\x2d\x40\x92\x98\ +\x60\xc3\x6a\x55\x84\x0d\xbd\xe0\x5e\x5f\xf0\x3e\x67\xa2\xe3\x5e\ +\x00\xb7\x2f\x48\xd8\x3c\x7c\x81\x92\x12\xed\xd8\x22\xdc\x65\xde\ +\x8c\x6c\x2e\x5d\x35\x81\xf4\xd4\x38\x94\x52\x20\xba\x9b\x9a\x61\ +\x08\xd3\xca\xd2\x19\x97\xff\x51\x67\xdd\xb4\x49\x99\xdc\x70\x59\ +\x19\x56\xab\x42\x29\x58\xfb\x4a\x2d\x0d\x2d\x6e\xfe\xf8\xdc\x7e\ +\x2a\xf6\x75\xc9\xdc\xe9\x39\x0a\xc0\x99\x60\xc7\x6a\x51\x98\xa6\ +\xd0\x37\x10\x20\x18\x32\x2e\x72\xd8\xad\x6f\x01\x0c\x44\x36\x1a\ +\x20\xc9\xe9\xc0\x6e\xb3\xf6\x01\x23\x83\x13\xfa\xdd\xc1\x8f\x00\ +\x7b\x18\xb0\x89\x89\x36\xac\x36\x0b\x49\x4e\xc7\xc8\x75\x8c\x06\ +\x3e\x40\xff\x40\xf4\xfd\x49\x4e\x3b\x36\x9b\x3a\x02\x23\x72\xd2\ +\x5c\x08\x27\x59\xdc\x1c\x91\x05\x16\xa1\xc9\xc5\xe9\x24\x3b\xf5\ +\xb8\xf1\xf7\x3e\x6c\x61\xcf\x81\x9e\x7b\x00\xde\xdc\xd0\x28\xf5\ +\x4d\x83\x84\x42\x87\x7f\x66\xe6\x94\x4c\xd2\x52\xb4\x82\xe9\xf6\ +\x04\x99\x39\x25\x93\x6b\x2f\x2e\xed\xbb\xf5\xfa\xa9\x77\xdf\x70\ +\xc5\xa4\x1d\x2b\x16\x14\x60\xb5\x2a\xf2\x73\x8e\x3e\x5f\xd7\x61\ +\xb7\x90\x9c\xe4\xa0\x64\x7c\x9a\xfa\xd2\xb5\x53\x99\x58\x94\x82\ +\x45\x29\x76\x55\x75\xb3\x7e\x4b\x2b\xc1\x90\x71\x11\xc0\xf4\xb2\ +\x0c\x52\x92\x1d\x23\xce\xb8\x8a\xaa\xae\xf2\x61\x53\xfd\x40\x9d\ +\x0b\x7f\xc0\x20\xce\x61\x65\x6a\x69\x3a\x39\x59\x89\x87\xc9\xaa\ +\x3d\xfb\x7b\xd8\xbd\xaf\x1b\x7f\x20\x7c\x57\x7d\x53\xbf\x6c\x8e\ +\xe8\x1e\x4a\xc1\x94\x12\x7d\xbd\x33\x27\x67\x8c\x5c\xc7\xd6\xdd\ +\x9d\x54\xd5\xf4\x8c\x78\x55\x37\xef\xd4\xef\xb7\x28\xc5\xcc\xc9\ +\x99\xa4\x24\xe9\xf7\x39\x6c\x16\x2c\x16\x08\x85\x4d\xea\x1a\xfb\ +\xa9\x6d\xec\x93\xae\x5e\x9f\xeb\x8c\xe1\x24\x1f\xe7\x5b\xba\x74\ +\xd5\x04\xca\x37\x34\xd1\x37\x18\x60\xdb\x9e\x4e\x7e\xf0\xab\xcd\ +\xdc\xfa\xed\xd7\xe5\xc7\x0f\x6c\xa5\xb9\xcd\xfd\x11\xde\xba\x7c\ +\x41\x3e\x8b\xe7\xe4\xd2\xd5\xe3\xe3\x60\x43\x3f\xbf\x7a\x64\x27\ +\x73\xa6\x65\xa5\x3b\x13\xed\x6b\x5c\xfd\x01\x1a\x5b\x06\x69\x6a\ +\x73\x33\xed\x08\x1d\xe3\x68\xb4\x7c\x41\x81\xfa\xd7\x9f\xac\x97\ +\xc6\xd6\x41\xfa\x06\x02\x3c\xff\x7a\x2d\xab\x96\x14\x96\x03\xea\ +\xea\x8b\x4b\xd8\xb8\xa3\x9d\xed\x7b\x3a\xf9\x70\x57\x07\xff\xeb\ +\x97\x9b\xb9\xfd\x3b\x6f\xc8\xbf\xff\x62\x23\x35\xf5\xfd\x28\x05\ +\x17\x2c\x2f\x62\xf6\xd4\xac\x8f\x1c\xb7\x6f\xc0\xcf\xaf\x1f\xdd\ +\xc5\x8b\xe5\xf5\x6b\xfa\xdd\x01\x76\x56\x76\x13\x08\x1a\x14\xe4\ +\x26\xb1\x72\x71\x21\xf9\x39\x49\xaa\xb5\xc3\x2d\x0b\x66\xe6\xd0\ +\xd5\xe3\xa3\xa9\x75\x90\x9f\xfd\x76\x3b\xb7\x7d\xfb\x75\xf9\xe7\ +\x1f\xbd\xcf\xce\xbd\x5d\x04\x43\x06\x93\x4b\xd2\x38\x7f\x59\x11\ +\xc3\xee\xfd\xf1\x85\x29\x38\x13\xec\xb8\xbd\x41\x1e\x79\xb2\x8a\ +\x17\xdf\x3a\x44\x76\x46\x42\xfa\xde\xea\x6e\x99\x35\x25\xfb\xf3\ +\xef\x27\x71\x7b\xa2\x3a\x82\xc3\x6e\x1d\xd1\x23\x96\x2f\x2c\x50\ +\x8f\x3c\xb9\x57\xfe\xeb\xcf\x7b\xa8\x6b\x1c\x60\xfb\x9e\x2e\xe2\ +\xe3\xac\xac\x5c\x58\x80\x69\xa6\x53\x73\xa8\x8f\xa1\x80\x81\x35\ +\xf2\xfe\xbc\xec\x24\xb5\x63\x6f\xa7\x24\xc4\xdb\x78\x63\x7d\x23\ +\x15\x55\xdd\x54\xd7\xf5\x61\xb3\x59\x08\x06\x0d\x4c\x84\xf1\x05\ +\xc9\x87\x99\x85\x49\x89\x76\xd2\x53\xe3\x08\x06\x0d\x12\xe2\x6d\ +\x58\x2c\xaa\x6f\xf8\x6f\xd7\x5f\x5a\xca\xf6\x3d\x9d\x54\xd7\xf7\ +\xd1\xd4\xea\x66\xcf\x7e\x7d\x47\x2f\x99\x9b\xaf\x5e\x28\xaf\x93\ +\x3f\x3c\x59\xc9\xb6\xdd\x9d\x6c\xdb\xdd\x89\xd5\x6a\x21\x14\x36\ +\x88\x8f\xb7\x71\xd5\x8a\x62\xbe\x76\xeb\xac\x91\x18\x8a\xc5\xa2\ +\x70\xd8\x2d\x38\x1c\x56\x96\xce\xcb\x47\x80\x4d\x3b\xdb\xe9\x1b\ +\x08\x60\x18\x42\x41\xae\x93\xaf\x7d\x79\x16\x8b\x66\xe5\xf6\x01\ +\x14\xe6\x25\xab\x2d\x15\xed\x12\xe7\xb0\x52\xbe\xb1\x99\x7d\xb5\ +\x2e\x6a\x0e\xe9\xf8\x92\xc5\xa2\x58\x34\x27\x97\x3b\x6f\x99\xc1\ +\xf9\xcb\xc6\x8d\x98\x69\x37\x5e\x31\x89\xe6\x76\x37\xdb\xf7\x74\ +\xd1\xdc\xee\xa6\xa9\xcd\x4d\x4e\x56\x02\x6e\x4f\x68\xcc\xf6\xed\ +\xa4\xa5\x2f\x1e\xa8\x75\xc9\x0f\x7e\xb5\x99\xf2\x0f\x9a\x18\xf2\ +\x87\xb9\xee\x92\x12\x7e\xf2\xdd\xe5\x94\x45\xbc\x84\x5e\x5f\xe8\ +\xbe\x4d\x3b\xda\xee\xa9\xd8\xd7\x8d\xdb\x13\xa2\x28\xdf\xc9\xc2\ +\xd9\xb9\x74\xf5\x0c\xd1\xd6\xe5\x21\x14\x32\x59\x36\x3f\x9f\xd9\ +\xd3\xa2\x77\x4b\x6d\x63\xbf\x54\xd7\xb9\xa8\x6b\x1c\xa4\xb7\x7f\ +\x08\x05\xa4\xa5\xc6\x53\x90\xeb\xa4\x28\x2f\x89\xd1\x71\x8d\x6d\ +\x7b\x3a\xa4\xab\xc7\x87\x69\x0a\xa5\x13\xd2\x98\x3e\xe9\x70\xff\ +\xc9\x86\x6d\x2d\xd2\x3f\x10\x04\x84\x49\xc5\xe9\x4c\x1d\xe5\x9a\ +\xaf\xa8\xea\x92\x7d\xb5\x2e\x9a\x5a\xdd\xda\xe3\x9a\x68\x67\xe2\ +\xb8\x54\xa6\x4f\xca\x38\x2c\xc8\x56\x59\xdd\x23\x15\xfb\xba\xf1\ +\xfa\x42\x94\x4e\x4c\x25\xc5\xe9\x60\x7f\x6d\x1f\x87\x5a\x06\x70\ +\xd8\xac\xcc\x98\x9c\xc1\xe2\x39\x79\xe4\xe7\x1e\xee\xa5\xad\xa9\ +\x77\xc9\xfe\x5a\x17\x75\x8d\x03\xf4\xbb\x83\x38\xec\x16\x8a\xf2\ +\x93\x99\x5a\x9a\xc6\xd2\x79\x1f\x8d\xcd\xec\xaa\xea\x92\xa6\xd6\ +\x41\xbc\xbe\x30\x22\x10\x17\x67\x65\xd5\x92\xc2\xbe\x9c\xcc\xc4\ +\x0c\x3e\xaf\xf4\x83\x5f\x6d\x96\x8b\x6f\x7d\x5e\xf2\x16\x3e\x2c\ +\x09\x93\x1f\x92\xd4\x99\xbf\x93\x07\x1f\xaf\x10\xc3\x30\x3f\x12\ +\xe1\xf3\xfa\x82\xf7\xb5\x77\x79\x8e\x49\x23\x33\x0c\xb3\xa4\xbd\ +\xcb\x2b\xed\x5d\x5e\x19\xd6\x27\xc6\xc4\x0a\x73\x07\x9e\x69\xef\ +\xf4\xc8\xa0\x27\xf0\xcc\xb1\x7c\xae\xa3\xcb\x2b\xae\xfe\xc3\x4d\ +\xe4\x8f\x0b\x33\xb4\x77\x79\xc6\x5c\xc7\x38\xed\x38\xc9\xa0\x27\ +\xf0\xcc\x17\xbe\xf6\xca\xea\x0f\x77\xb6\x23\xa2\xdd\xcd\xd7\x5d\ +\x5a\xc2\xb7\xbf\x32\x8f\x69\x93\x32\xff\x2e\x13\xb1\x3f\x6f\x74\ +\x52\x36\xe9\x07\xbf\xdc\x24\x8d\xad\x6e\x72\xb3\x13\x99\x3d\x35\ +\x8b\xc5\x73\x72\x89\x55\xa4\xf5\x2c\x9d\x21\x20\x69\x68\x19\x90\ +\x60\xd0\x24\x21\xde\xf6\xa9\x49\x46\x67\xe9\x2c\x9d\xa5\xcf\x21\ +\xfd\x7f\x20\x92\x48\xac\x58\x09\xb6\x48\x00\x00\x00\x00\x49\x45\ +\x4e\x44\xae\x42\x60\x82\ +\x00\x00\x07\x4d\ +\x89\ +\x50\x4e\x47\x0d\x0a\x1a\x0a\x00\x00\x00\x0d\x49\x48\x44\x52\x00\ +\x00\x00\x19\x00\x00\x00\x1c\x08\x06\x00\x00\x00\x94\x24\x14\xd0\ +\x00\x00\x00\x01\x73\x52\x47\x42\x00\xae\xce\x1c\xe9\x00\x00\x00\ +\x06\x62\x4b\x47\x44\x00\xff\x00\xff\x00\xff\xa0\xbd\xa7\x93\x00\ +\x00\x00\x09\x70\x48\x59\x73\x00\x00\x0b\x13\x00\x00\x0b\x13\x01\ +\x00\x9a\x9c\x18\x00\x00\x00\x07\x74\x49\x4d\x45\x07\xdc\x02\x09\ +\x11\x34\x07\x96\x4a\xc5\x0d\x00\x00\x06\xcd\x49\x44\x41\x54\x48\ +\xc7\x9d\x56\xdd\x6b\x5c\xc7\x15\xff\x9d\x99\xbb\xbb\x77\x3f\x2c\ +\x69\xb5\xbb\xfa\x8e\xd5\xae\x8d\x9b\x90\x50\xd7\x34\x36\xa9\x0a\ +\xaa\x93\x42\x5a\x1c\xda\xa6\x26\x84\x42\x1b\x30\xfd\xc0\x2d\xce\ +\x83\xdf\xfa\x37\x34\xaf\x21\x50\x3f\xf4\xa1\x06\x63\x1b\x5a\x63\ +\x28\x7d\x29\xc6\x54\xea\x83\xeb\xa6\xd4\x72\xb2\x51\x2c\x59\x76\ +\x24\xaf\xfc\x6d\xaf\xb4\xba\xab\xfb\x35\x33\xe7\xf4\x61\x57\x4a\ +\xdc\xba\x24\xf4\xc0\x70\x99\xb9\xc3\xf9\xcd\x6f\xce\x99\xdf\x39\ +\xc0\x53\x2c\x0c\x43\xfc\x3f\xd6\xe9\x74\x9e\xba\xae\x9e\xb6\x28\ +\x22\x4f\xdd\x7c\xfb\xf6\x6d\x2f\x0c\xc3\x3c\x33\xfb\x49\x92\xd0\ +\x17\x05\x7f\x62\x63\x14\x45\x20\x22\xf8\xbe\x0f\x00\x68\x36\x9b\ +\xa8\xd5\x6a\x15\x00\x35\x00\x65\x00\x03\x44\xe4\x03\x60\x11\xe9\ +\x88\xc8\x63\xe7\xdc\x63\xe7\xdc\x7a\x7f\x7f\x7f\xd0\x3b\x08\xc6\ +\xc7\xc7\x21\x22\x20\xa2\x27\x41\xc2\x30\x44\xa1\x50\xf8\x2c\x75\ +\x4d\x44\x63\x5a\xeb\xd7\x88\x68\x9a\x88\x26\x00\xf4\x01\xc8\x32\ +\xb3\x10\x51\xc8\xcc\x2d\x11\xf9\x88\x99\xff\x0a\xe0\x42\xa9\x54\ +\x0a\xb7\x6e\x82\x88\x70\xe2\xc4\x09\x1c\x3d\x7a\x14\xde\x36\x25\ +\xfa\x94\x54\x92\x24\x35\x11\x79\x4d\x44\x7e\x42\x44\xbb\x94\x52\ +\x7d\x4a\x29\x5f\x44\x3c\x11\x21\xa5\x14\x44\xc4\x69\xad\x2d\x11\ +\xed\x15\x91\x97\x8d\x31\x6f\x76\x3a\x9d\xf7\x4a\xa5\xd2\xa5\x6d\ +\x06\x5b\xdf\x24\x49\x00\x00\xb9\x5c\x0e\x00\x90\xa6\xe9\x84\xb5\ +\xf6\xfb\x00\xde\x52\x4a\xbd\xa4\x94\x82\x52\x0a\x44\x0a\x22\x0c\ +\x11\xd9\x1e\x44\xb4\xed\xc8\x18\x93\x88\xc8\x59\xad\xf5\xef\x72\ +\xb9\xdc\xdf\x88\x68\x3b\xb0\xde\x96\xf3\x1e\xcd\x42\x14\x45\x6f\ +\x32\xf3\x4f\x89\xe8\xf9\x9e\x03\x11\x11\xea\x86\x41\x20\x22\x24\ +\x22\x60\xe6\xad\xd3\x8a\x52\x0a\x5a\xeb\x5c\x9a\xa6\x6f\x05\x41\ +\x30\x7a\xea\xd4\xa9\xb7\x01\xdc\x00\xe0\x00\x40\x7f\x06\xa0\x0f\ +\xc0\xab\x51\x14\xbd\x2d\x22\x5f\x55\x4a\xfd\x47\x82\x74\x1d\x77\ +\x01\xf1\x04\xa3\xde\x4d\x50\x14\x45\xb4\xb0\xb0\xd0\x7f\xe9\xd2\ +\xa5\x6a\xb5\x5a\x5d\x5c\x5c\x5c\x7c\x04\xe0\xd3\x98\xcc\xcc\xce\ +\x7e\xb9\x5a\xa9\xfe\x60\x72\x72\xe7\xae\x52\xa9\x08\x00\xd2\xe9\ +\x74\x48\x29\x45\x4a\x2b\x11\x66\x12\x01\x20\x80\x48\x17\x44\x29\ +\x25\xcc\x42\x41\xb0\x81\x46\xa3\x21\xf3\xf3\xf3\x34\x37\x37\x57\ +\x9d\x9b\x9b\x3b\xb4\xb8\xb8\x38\x0b\x60\x15\x40\xb0\xc5\x24\xef\ +\x69\xef\x9b\xcd\xd5\xd5\x9f\x75\x3a\x41\x35\x4d\x13\x05\x08\x32\ +\x99\x2c\xb4\xd6\x24\xec\xc8\x98\x14\xec\x2c\x8c\x49\x91\x24\x31\ +\xe2\x38\x46\x1c\xc7\xd4\x6e\xb7\xb1\xbc\xfc\x09\xce\x9f\x3f\x4f\ +\x2b\x2b\x2b\xa2\x94\x22\xcf\xf3\x76\x24\x49\x72\xf7\xce\x9d\x3b\ +\xcb\x00\xee\x69\x00\x28\x16\x0b\xe3\x1f\x7d\xd8\xf8\xf6\x3f\x2e\ +\x5f\xfe\xe1\xc5\x8b\x17\xd4\xc2\xc2\x02\x44\x84\x26\x27\x77\x92\ +\xd6\x1a\x71\x1c\x22\x89\x23\xa4\x69\x82\x4e\x27\x40\xbb\xbd\x8e\ +\x76\x7b\x1d\xf7\xef\xdd\xc7\xca\xf2\x32\xae\x7e\x30\x87\xd3\xa7\ +\xcf\x60\x6a\x6a\x0a\xc7\x8f\x1f\xa7\x7d\xfb\xf6\x21\x08\x02\xd7\ +\x6e\xb7\x6f\x3d\x78\xf0\xe0\xaa\x07\x00\x26\x35\xcf\x40\xa4\x0e\ +\x02\xe2\x24\xc4\xcc\xcc\xac\x34\x1a\xf3\x74\xf6\xec\x59\xbc\xf8\ +\xe2\xd7\xb1\x77\xef\x0b\x18\x1b\x1f\x85\x73\x0e\x26\xb5\xb0\xd6\ +\xc1\x39\x06\x11\x61\x63\xa3\x8d\xf5\xb5\x35\xf8\xbe\x8f\x52\xa9\ +\x84\x4a\xa5\x82\xa1\xa1\x21\x1c\x38\x70\xe0\x39\xcf\xf3\x76\x37\ +\x1a\x0d\xf2\x7a\x19\x32\xc2\x22\x63\xce\x39\x08\x0b\x92\x74\x03\ +\xeb\xed\x0d\xdc\xb9\xbb\x8a\xd2\x8e\x02\x86\x86\x07\xa1\x33\xd4\ +\x0d\x3c\x03\x10\x82\x63\x07\x02\x41\x84\x51\xad\x0e\xca\xc4\xc4\ +\x38\x5a\xad\x16\x5d\xb9\x72\x05\xe3\xe3\xe3\x00\xd0\x5f\x2e\x97\ +\x87\x00\x28\x4f\x44\xf0\xcc\xc4\xc4\x8e\x24\x89\x07\xa2\x28\x82\ +\xb5\x06\xce\x59\xca\xe5\x72\xf2\xa5\xc9\x49\x8c\x8e\x8c\x50\x26\ +\x93\x41\x10\x04\x20\x60\xfb\xcd\x58\xeb\x10\x6c\x04\xc8\xf9\x3e\ +\x06\xca\xfd\x54\xa9\x54\x70\xf3\xe6\x0d\x9c\x3b\x77\x4e\xea\xf5\ +\x3a\x5a\xad\x16\x85\x61\xe8\x1f\x39\x72\xa4\xcb\xe4\x95\x83\x07\ +\xbd\xcd\xcd\x8e\xb7\xb6\xd6\x42\x67\x33\x40\x14\x45\x50\x4a\x63\ +\x78\x68\x08\xec\x18\x8f\x1e\x3c\x84\xef\x67\x01\x08\xb4\xf6\xa0\ +\xb5\x07\xeb\x1c\x56\x96\x6f\x21\x8e\x63\x14\x0a\x05\xb4\x5a\x8f\ +\x11\xc7\x29\x6e\xdf\xbe\x83\x99\x99\x19\xec\xd9\xb3\x07\x95\x4a\ +\x05\x87\x0e\x1d\x22\xcf\xc4\xc0\xc8\xf0\x50\x6c\xcc\x40\x54\xab\ +\x55\x90\xc4\x11\x8c\xb1\x20\x22\xca\x17\xf2\x28\x15\x0b\x60\xc7\ +\x48\x93\x14\x2c\x02\xe1\x18\x4a\x69\x64\x32\x59\x98\x34\xc5\x6a\ +\xb3\x89\xb5\xf5\x0d\xc9\xfa\x05\x94\xcb\x83\xa4\x94\xa2\x66\xb3\ +\x89\x7a\xbd\x8e\x5c\x2e\x27\xf5\x7a\x5d\x54\x36\x4f\xc8\x66\x33\ +\xad\x42\x21\xff\x68\x70\xa0\x8c\xa1\xda\x30\xc6\xc6\xc6\x64\x6c\ +\x6c\x14\x83\x83\x65\xf8\x7e\x0e\x22\xdc\x0d\xb6\x75\x48\x53\x8b\ +\x38\x8a\x11\x45\x09\x9c\x75\xd8\x68\x07\x68\x36\x9b\x94\x26\x09\ +\x65\x32\x19\xf1\xfd\x3c\x7c\xdf\x87\xe7\x79\xf0\x3c\x2f\xa8\xd7\ +\xeb\xec\x01\x00\x33\xaf\x3a\xe7\x96\x99\x19\xdd\x61\x49\x20\xf0\ +\x3c\x0f\x80\x40\x84\xb1\xa5\x9f\x22\x04\xe7\x04\x41\xb0\x89\x8d\ +\x8d\x0e\x58\x04\xc5\x42\x01\xc5\x42\x11\xe1\x66\x88\xc4\x18\xf4\ +\xf5\xf5\xc1\x39\xb7\x12\x86\x61\xb3\x5c\x2e\x77\x41\x04\x74\xd7\ +\x32\xdf\x60\xe1\x8e\xb0\x2d\x49\x37\x85\x60\x4c\xda\x03\x71\x60\ +\xb6\x3d\x91\x24\x30\x0b\x2c\x3b\x14\x8a\x79\x4c\x4e\xee\xc4\xd8\ +\xd8\x28\x32\x7e\x11\xcc\x40\xa9\x27\xb6\xf9\x7c\x7e\xce\x39\xd7\ +\x20\xa2\x6e\x65\x7c\xf9\x95\x1f\x3f\x34\xce\xcd\x1b\x6b\x3e\x4e\ +\x8d\x49\x8d\xb1\x30\xa9\x91\x24\x89\xd1\x1b\x12\x6f\xbf\xf2\x08\ +\x71\x1c\xc3\xa4\x06\x4a\x29\x14\x8a\x45\xf4\x0f\x0c\x48\x2e\x9b\ +\x93\x6c\x36\x43\xbe\xef\xa3\x58\x2c\xc6\x5a\xeb\xf7\x89\xe8\xe3\ +\x6d\xed\x7a\xf5\xbb\x5f\x93\x5f\xfd\xf2\x17\xd7\x98\xf9\x2f\xd6\ +\x98\x71\x66\x37\x26\xec\xc8\x39\x27\x00\x43\xc0\x24\xe2\x20\x4c\ +\x10\x01\x98\x05\x5d\x5d\x24\x88\x88\x30\x40\x10\x0d\xd2\x1a\xda\ +\xa3\x24\x4d\xd3\x06\x33\xff\x73\x6a\x6a\xea\xd6\x13\x95\xf1\xca\ +\xbf\xee\xe3\xdd\x77\x7f\xbd\x33\x0c\x83\xdf\x3a\x6b\xbf\x63\xad\ +\x55\xe2\x5c\x57\xde\xc1\x24\xc2\x3d\xe7\xd2\x8b\x1b\x41\x91\x02\ +\x88\x44\x00\x12\x68\x64\xb3\x59\x64\xb2\x99\x65\xcf\xf3\xde\xd1\ +\x5a\x9f\x3b\x73\xe6\xcc\x7d\x00\xf0\xfe\xf8\x87\x77\xb0\x7f\xff\ +\x27\xd8\x39\x39\x0c\x00\xb7\xde\x78\xe3\xf5\xdf\x84\x61\xd4\x4e\ +\xd3\xf4\x47\xce\x18\x12\x74\xa5\x57\xd8\x81\x9d\x80\x99\x85\x45\ +\x00\x51\x44\x5d\x10\x02\x08\xa4\x35\xac\xb5\x0b\xbe\xf8\x67\xad\ +\xb5\x7f\xb6\xd6\x3e\x00\x80\x63\xc7\x8e\x41\x17\xf2\xe3\xb8\x78\ +\xd1\xe0\xea\xd5\xab\x00\x80\xf9\xf9\x6b\x2b\xc3\xc3\xc3\x2d\x6b\ +\x6d\x39\x4e\xa2\x6c\x9c\x24\x99\x24\x8a\x75\x1c\xc5\x2a\x8e\x13\ +\x44\x51\x4c\x71\x9c\x90\x31\x16\x69\x62\x60\x52\x9b\x1a\x6b\xdb\ +\x22\x58\xb2\x6c\xcf\x18\x63\x4e\x5f\xb8\x70\xe1\xe6\xd2\xd2\x12\ +\x0e\x1f\x3e\x8c\x93\x27\x4f\x7e\x7a\x5d\xdf\x9a\x9e\xc6\xcc\xec\ +\xec\x76\x95\x7a\xf6\xd9\x3d\x95\x6c\x26\xf3\xf3\x24\x8a\x0e\x1a\ +\x63\xbe\x22\xcc\x35\x11\x14\x9c\x63\x25\x22\xf0\xbc\xac\x23\xa8\ +\x35\x90\x5a\xd5\x9e\xfa\x00\x0a\xa7\x49\xd1\xe5\xeb\xd7\xaf\xaf\ +\x01\xc0\xf4\xf4\x34\x66\x7b\xfe\xfe\xab\x91\xf8\xd3\xf9\xf7\xf0\ +\xbd\xd7\x8f\xe1\xda\xb5\xc5\xc7\x2f\x3c\xff\xdc\xef\xd3\xd4\x5c\ +\xb4\xc6\x4c\x8a\xf0\x84\x73\xdc\xcf\x2c\x39\x40\x31\xbb\x34\x22\ +\xd2\x0f\x95\xd2\xcb\x20\xbd\x14\x47\xf1\xfd\xbb\xf7\xee\x6e\x7e\ +\x6e\xdf\xb5\x65\xdf\x78\x69\x3f\x2e\xfd\xfd\xfd\xed\x79\xce\xcb\ +\xe6\x8b\x05\x7f\x80\xd9\xe6\x01\x9d\x21\xd2\xc2\x4e\x12\x02\x75\ +\xd6\xc3\xf5\x35\x00\xbc\xb5\x77\xf7\xae\xdd\x20\x22\x5c\x5f\xba\ +\xfe\xf9\x5d\xdf\x48\xad\x86\x91\x5a\xed\x0b\x75\x88\xa3\xd5\x51\ +\x8c\x56\x47\xff\xe7\xff\x7f\x03\x97\x54\x18\xa6\xd8\x35\x20\x89\ +\x00\x00\x00\x00\x49\x45\x4e\x44\xae\x42\x60\x82\ +" + +qt_resource_name = b"\ +\x00\x06\ +\x07\x03\x7d\xc3\ +\x00\x69\ +\x00\x6d\x00\x61\x00\x67\x00\x65\x00\x73\ +\x00\x08\ +\x02\x8c\x59\xa7\ +\x00\x70\ +\x00\x6c\x00\x61\x00\x79\x00\x2e\x00\x70\x00\x6e\x00\x67\ +\x00\x08\ +\x0b\x63\x58\x07\ +\x00\x73\ +\x00\x74\x00\x6f\x00\x70\x00\x2e\x00\x70\x00\x6e\x00\x67\ +\x00\x0c\ +\x07\x72\xc5\xe7\ +\x00\x6a\ +\x00\x64\x00\x65\x00\x72\x00\x6f\x00\x62\x00\x6f\x00\x74\x00\x2e\x00\x70\x00\x6e\x00\x67\ +\x00\x08\ +\x08\x2f\x5a\x47\ +\x00\x62\ +\x00\x61\x00\x6c\x00\x6c\x00\x2e\x00\x70\x00\x6e\x00\x67\ +" + +qt_resource_struct = b"\ +\x00\x00\x00\x00\x00\x02\x00\x00\x00\x01\x00\x00\x00\x01\ +\x00\x00\x00\x00\x00\x02\x00\x00\x00\x04\x00\x00\x00\x02\ +\x00\x00\x00\x12\x00\x00\x00\x00\x00\x01\x00\x00\x00\x00\ +\x00\x00\x00\x3e\x00\x00\x00\x00\x00\x01\x00\x00\x27\x30\ +\x00\x00\x00\x5c\x00\x00\x00\x00\x00\x01\x00\x00\x5f\xcb\ +\x00\x00\x00\x28\x00\x00\x00\x00\x00\x01\x00\x00\x14\x5b\ +" + +def qInitResources(): + QtCore.qRegisterResourceData(0x01, qt_resource_struct, qt_resource_name, qt_resource_data) + +def qCleanupResources(): + QtCore.qUnregisterResourceData(0x01, qt_resource_struct, qt_resource_name, qt_resource_data) + +qInitResources()