Merge pull request #6 from SemvdH/camera
Camera node to take pictures is done
This commit is contained in:
BIN
drone_image.jpg
Normal file
BIN
drone_image.jpg
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 118 KiB |
0
drone_scripts/TerabeeLog.log
Normal file
0
drone_scripts/TerabeeLog.log
Normal file
32
drone_scripts/find_id.sh
Executable file
32
drone_scripts/find_id.sh
Executable file
@@ -0,0 +1,32 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
lidarserial=00000000001A
|
||||||
|
heightserial=369A378F3439
|
||||||
|
|
||||||
|
find_by_id(){
|
||||||
|
v=${1%:*}; p=${1#*:} # split vid:pid into 2 vars
|
||||||
|
v=${v#${v%%[!0]*}}; p=${p#${p%%[!0]*}} # strip leading zeros
|
||||||
|
grep -il "^PRODUCT=$v/$p" /sys/bus/usb/devices/*:*/uevent |
|
||||||
|
sed s,uevent,, |
|
||||||
|
xargs -r grep -r '^DEVNAME=' --include uevent
|
||||||
|
}
|
||||||
|
ids=$(find_by_id 0483:5740)
|
||||||
|
echo $ids
|
||||||
|
|
||||||
|
for device in $ids
|
||||||
|
do
|
||||||
|
id=$(echo $device | cut -d "=" -f 2)
|
||||||
|
serial=$(/bin/udevadm info --name=/dev/$id | grep SERIAL_SHORT | cut -d = -f 2)
|
||||||
|
echo "device "$id" has serial number "$serial
|
||||||
|
if [ $serial = $lidarserial ]
|
||||||
|
then
|
||||||
|
echo "LIDAR on port" $id
|
||||||
|
echo $id > /home/ubuntu/drone_conf/lidar.conf
|
||||||
|
elif [ $serial = $heightserial ]
|
||||||
|
then
|
||||||
|
echo "height sensor on port" $id
|
||||||
|
echo $id > /home/ubuntu/drone_conf/height.conf
|
||||||
|
else
|
||||||
|
echo "unknown serial number" $serial
|
||||||
|
fi
|
||||||
|
done
|
||||||
10
drone_scripts/start_height_sensor.sh
Executable file
10
drone_scripts/start_height_sensor.sh
Executable file
@@ -0,0 +1,10 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
. /home/ubuntu/source_ros2.sh
|
||||||
|
|
||||||
|
SERIAL=$(cat /home/ubuntu/drone_conf/height.conf)
|
||||||
|
echo "serial port is "$SERIAL
|
||||||
|
echo ros2 run height height_reader --ros-args -p height_serial_port:=/dev/${SERIAL}
|
||||||
|
ros2 run height height_reader --ros-args -p height_serial_port:=/dev/${SERIAL} | tee /home/ubuntu/drone_log/height.log
|
||||||
|
|
||||||
|
|
||||||
10
drone_scripts/start_lidar.sh
Executable file
10
drone_scripts/start_lidar.sh
Executable file
@@ -0,0 +1,10 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
. /home/ubuntu/source_ros2.sh
|
||||||
|
|
||||||
|
SERIAL=$(cat /home/ubuntu/drone_conf/lidar.conf)
|
||||||
|
echo "serial port is "$SERIAL
|
||||||
|
echo ros2 run object_detection lidar_reader --ros-args -p lidar_serial_port:=/dev/${SERIAL}
|
||||||
|
ros2 run object_detection lidar_reader --ros-args -p lidar_serial_port:=/dev/${SERIAL} | tee /home/ubuntu/drone_log/lidar.log
|
||||||
|
|
||||||
|
|
||||||
1
get_usb_device_port.sh
Normal file
1
get_usb_device_port.sh
Normal file
@@ -0,0 +1 @@
|
|||||||
|
/bin/udevadm info --name=/dev/ttyACM0 | grep SERIAL_SHORT | cut -d = -f 2
|
||||||
12
services/drone_find_usb_devices.service
Normal file
12
services/drone_find_usb_devices.service
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
[Unit]
|
||||||
|
Description=Service to start the script that sets the serial ports for the USB sensors
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
Type=simple
|
||||||
|
Restart=on-failure
|
||||||
|
User=ubuntu
|
||||||
|
ExecStart=/home/ubuntu/drone_scripts/find_id.sh
|
||||||
|
WorkingDirectory=/home/ubuntu
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
14
services/drone_height_sensor.service
Normal file
14
services/drone_height_sensor.service
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
[Unit]
|
||||||
|
Description=Height sensor for drone running in ROS 2
|
||||||
|
After=drone_find_usb_devices.service
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
Type=simple
|
||||||
|
Restart=on-failure
|
||||||
|
User=ubuntu
|
||||||
|
ExecStart=/home/ubuntu/drone_scripts/start_height_sensor.sh
|
||||||
|
Environment="HOME=root"
|
||||||
|
WorkingDirectory=/home/ubuntu
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
14
services/drone_lidar.service
Normal file
14
services/drone_lidar.service
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
[Unit]
|
||||||
|
Description=Height sensor for drone running in ROS 2
|
||||||
|
After=drone_find_usb_devices.service
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
Type=simple
|
||||||
|
Restart=on-failure
|
||||||
|
User=ubuntu
|
||||||
|
ExecStart=/home/ubuntu/drone_scripts/start_lidar.sh
|
||||||
|
Environment="HOME=root"
|
||||||
|
WorkingDirectory=/home/ubuntu
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
0
src/camera/camera/__init__.py
Normal file
0
src/camera/camera/__init__.py
Normal file
50
src/camera/camera/camera_controller.py
Normal file
50
src/camera/camera/camera_controller.py
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
import rclpy
|
||||||
|
from rclpy.node import Node
|
||||||
|
|
||||||
|
from drone_services.srv import TakePicture
|
||||||
|
import os
|
||||||
|
from datetime import datetime
|
||||||
|
|
||||||
|
# import cv2
|
||||||
|
|
||||||
|
RES_4K_H = 3496
|
||||||
|
RES_4K_W = 4656
|
||||||
|
|
||||||
|
|
||||||
|
class CameraController(Node):
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__('camera_controller')
|
||||||
|
|
||||||
|
self.get_logger().info("Camera controller started. Waiting for service call...")
|
||||||
|
self.srv = self.create_service(
|
||||||
|
TakePicture, 'drone/picture', self.take_picture_callback)
|
||||||
|
|
||||||
|
def take_picture_callback(self, request, response):
|
||||||
|
if (request.input_name == "default"):
|
||||||
|
self.get_logger().info("Taking picture with default filename")
|
||||||
|
now = datetime.now().strftime("droneimage_%Y-%m-%d_%H-%M-%S")
|
||||||
|
imagename = "/home/ubuntu/drone_img/" + now + ".jpg"
|
||||||
|
response.filename = imagename
|
||||||
|
else:
|
||||||
|
response.filename = request.input_name
|
||||||
|
os.system('fswebcam -r ' + RES_4K_W + 'x' + RES_4K_H + ' ' + response.filename)
|
||||||
|
self.get_logger().info("Picture saved as " + response.filename)
|
||||||
|
|
||||||
|
return response
|
||||||
|
|
||||||
|
|
||||||
|
def main(args=None):
|
||||||
|
rclpy.init(args=args)
|
||||||
|
|
||||||
|
test_controller = CameraController()
|
||||||
|
rclpy.spin(test_controller)
|
||||||
|
|
||||||
|
# Destroy the node explicitly
|
||||||
|
# (optional - otherwise it will be done automatically
|
||||||
|
# when the garbage collector destroys the node object)
|
||||||
|
test_controller.destroy_node()
|
||||||
|
rclpy.shutdown()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
||||||
21
src/camera/package.xml
Normal file
21
src/camera/package.xml
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
<?xml version="1.0"?>
|
||||||
|
<?xml-model href="http://download.ros.org/schema/package_format3.xsd" schematypens="http://www.w3.org/2001/XMLSchema"?>
|
||||||
|
<package format="3">
|
||||||
|
<name>camera</name>
|
||||||
|
<version>0.0.0</version>
|
||||||
|
<description>Package for controlling the camera of the drone</description>
|
||||||
|
<maintainer email="semmer99@gmail.com">ubuntu</maintainer>
|
||||||
|
<license>Apache License 2.0</license>
|
||||||
|
|
||||||
|
<exec_depend>rclpy</exec_depend>
|
||||||
|
<exec_depend>drone_services</exec_depend>
|
||||||
|
|
||||||
|
<test_depend>ament_copyright</test_depend>
|
||||||
|
<test_depend>ament_flake8</test_depend>
|
||||||
|
<test_depend>ament_pep257</test_depend>
|
||||||
|
<test_depend>python3-pytest</test_depend>
|
||||||
|
|
||||||
|
<export>
|
||||||
|
<build_type>ament_python</build_type>
|
||||||
|
</export>
|
||||||
|
</package>
|
||||||
0
src/camera/resource/camera
Normal file
0
src/camera/resource/camera
Normal file
4
src/camera/setup.cfg
Normal file
4
src/camera/setup.cfg
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
[develop]
|
||||||
|
script-dir=$base/lib/camera
|
||||||
|
[install]
|
||||||
|
install-scripts=$base/lib/camera
|
||||||
26
src/camera/setup.py
Normal file
26
src/camera/setup.py
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
from setuptools import setup
|
||||||
|
|
||||||
|
package_name = 'camera'
|
||||||
|
|
||||||
|
setup(
|
||||||
|
name=package_name,
|
||||||
|
version='0.0.0',
|
||||||
|
packages=[package_name],
|
||||||
|
data_files=[
|
||||||
|
('share/ament_index/resource_index/packages',
|
||||||
|
['resource/' + package_name]),
|
||||||
|
('share/' + package_name, ['package.xml']),
|
||||||
|
],
|
||||||
|
install_requires=['setuptools'],
|
||||||
|
zip_safe=True,
|
||||||
|
maintainer='ubuntu',
|
||||||
|
maintainer_email='semmer99@gmail.com',
|
||||||
|
description='Package for controlling the camera of the drone',
|
||||||
|
license='Apache License 2.0',
|
||||||
|
tests_require=['pytest'],
|
||||||
|
entry_points={
|
||||||
|
'console_scripts': [
|
||||||
|
'camera_controller = camera.camera_controller:main'
|
||||||
|
],
|
||||||
|
},
|
||||||
|
)
|
||||||
23
src/camera/test/test_copyright.py
Normal file
23
src/camera/test/test_copyright.py
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
# Copyright 2015 Open Source Robotics Foundation, Inc.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
from ament_copyright.main import main
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.copyright
|
||||||
|
@pytest.mark.linter
|
||||||
|
def test_copyright():
|
||||||
|
rc = main(argv=['.', 'test'])
|
||||||
|
assert rc == 0, 'Found errors'
|
||||||
25
src/camera/test/test_flake8.py
Normal file
25
src/camera/test/test_flake8.py
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
# Copyright 2017 Open Source Robotics Foundation, Inc.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
from ament_flake8.main import main_with_errors
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.flake8
|
||||||
|
@pytest.mark.linter
|
||||||
|
def test_flake8():
|
||||||
|
rc, errors = main_with_errors(argv=[])
|
||||||
|
assert rc == 0, \
|
||||||
|
'Found %d code style errors / warnings:\n' % len(errors) + \
|
||||||
|
'\n'.join(errors)
|
||||||
23
src/camera/test/test_pep257.py
Normal file
23
src/camera/test/test_pep257.py
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
# Copyright 2015 Open Source Robotics Foundation, Inc.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
from ament_pep257.main import main
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.linter
|
||||||
|
@pytest.mark.pep257
|
||||||
|
def test_pep257():
|
||||||
|
rc = main(argv=['.', 'test'])
|
||||||
|
assert rc == 0, 'Found code style errors / warnings'
|
||||||
@@ -23,6 +23,8 @@ find_package(rosidl_default_generators REQUIRED)
|
|||||||
rosidl_generate_interfaces(${PROJECT_NAME}
|
rosidl_generate_interfaces(${PROJECT_NAME}
|
||||||
"srv/SetAttitude.srv"
|
"srv/SetAttitude.srv"
|
||||||
"srv/SetVelocity.srv"
|
"srv/SetVelocity.srv"
|
||||||
|
"srv/TakePicture.srv"
|
||||||
|
"srv/SetVehicleControl.srv"
|
||||||
)
|
)
|
||||||
|
|
||||||
if(BUILD_TESTING)
|
if(BUILD_TESTING)
|
||||||
|
|||||||
3
src/drone_services/srv/SetVehicleControl.srv
Normal file
3
src/drone_services/srv/SetVehicleControl.srv
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
int32 control # control bitmask
|
||||||
|
---
|
||||||
|
int8 status # status of operation
|
||||||
3
src/drone_services/srv/TakePicture.srv
Normal file
3
src/drone_services/srv/TakePicture.srv
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
wstring input_name "default" # name of the input file
|
||||||
|
---
|
||||||
|
wstring filename # output file name
|
||||||
@@ -1,8 +1,4 @@
|
|||||||
#include <chrono>
|
#include <chrono>
|
||||||
#include <functional>
|
|
||||||
#include <memory>
|
|
||||||
#include <string>
|
|
||||||
#include <iostream>
|
|
||||||
|
|
||||||
#include "rclcpp/rclcpp.hpp"
|
#include "rclcpp/rclcpp.hpp"
|
||||||
#include "object_detection/msg/lidar_reading.hpp"
|
#include "object_detection/msg/lidar_reading.hpp"
|
||||||
|
|||||||
@@ -1,8 +1,4 @@
|
|||||||
#include <chrono>
|
#include <chrono>
|
||||||
#include <functional>
|
|
||||||
#include <memory>
|
|
||||||
#include <string>
|
|
||||||
#include <iostream>
|
|
||||||
|
|
||||||
#include "rclcpp/rclcpp.hpp"
|
#include "rclcpp/rclcpp.hpp"
|
||||||
#include "object_detection/msg/multiflex_reading.hpp"
|
#include "object_detection/msg/multiflex_reading.hpp"
|
||||||
@@ -11,6 +7,8 @@
|
|||||||
#include <terabee/ITerarangerMultiflex.hpp>
|
#include <terabee/ITerarangerMultiflex.hpp>
|
||||||
#include <terabee/DistanceData.hpp>
|
#include <terabee/DistanceData.hpp>
|
||||||
|
|
||||||
|
// #include <iostream>
|
||||||
|
|
||||||
using terabee::DistanceData;
|
using terabee::DistanceData;
|
||||||
|
|
||||||
using namespace std::chrono_literals;
|
using namespace std::chrono_literals;
|
||||||
@@ -42,9 +40,9 @@ public:
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!multiflex->configureNumberOfSensors(0x3f)) // check if all 6 distance sensors work
|
if (!multiflex->configureNumberOfSensors(0x7)) // check if all 6 distance sensors work
|
||||||
{
|
{
|
||||||
RCLCPP_ERROR(this->get_logger(), "Failed to set the number of sensors to 6!");
|
RCLCPP_ERROR(this->get_logger(), "Failed to set the number of sensors to 3!");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -76,7 +74,8 @@ private:
|
|||||||
auto msg = object_detection::msg::MultiflexReading();
|
auto msg = object_detection::msg::MultiflexReading();
|
||||||
for (size_t i = 0; i < data.size(); i++)
|
for (size_t i = 0; i < data.size(); i++)
|
||||||
{
|
{
|
||||||
msg.distance_data[i] = data.distance[i];
|
RCLCPP_INFO(this->get_logger(), "distance %f", data.distance[i]);
|
||||||
|
// msg.distance_data[i] = data.distance[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
// publish message
|
// publish message
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ class TestController(Node):
|
|||||||
self.get_logger().info('service not available, waiting again...')
|
self.get_logger().info('service not available, waiting again...')
|
||||||
self.req = SetAttitude.Request()
|
self.req = SetAttitude.Request()
|
||||||
|
|
||||||
self.get_logger().info("Controls:\nW - forward\nS - backward\nA - left\nD - right\nQ - rotate left\nE - rotate right\nSpace - up\nShift - down\nN - emergency stop\nEsc - exit")
|
self.get_logger().info("Controls:\nW - forward\nS - backward\nA - left\nD - right\nQ - rotate left\nE - rotate right\nSpace - up\nZ - down\nV - Down nudge\nF - Up nudge\nN - emergency stop\nEsc - exit")
|
||||||
|
|
||||||
def spin(self):
|
def spin(self):
|
||||||
while rclpy.ok():
|
while rclpy.ok():
|
||||||
@@ -43,27 +43,27 @@ class TestController(Node):
|
|||||||
# self.get_logger().info('pressed ' + char)
|
# self.get_logger().info('pressed ' + char)
|
||||||
if key == 'w':
|
if key == 'w':
|
||||||
self.get_logger().info('forward')
|
self.get_logger().info('forward')
|
||||||
self.send_request(pitch=-10.0, yaw=0.0,
|
self.send_request(pitch=-1.0, yaw=0.0,
|
||||||
roll=0.0, thrust=0.0)
|
roll=0.0, thrust=0.0)
|
||||||
if key == 's':
|
if key == 's':
|
||||||
self.get_logger().info('backward')
|
self.get_logger().info('backward')
|
||||||
self.send_request(pitch=10.0, yaw=0.0,
|
self.send_request(pitch=1.0, yaw=0.0,
|
||||||
roll=0.0, thrust=0.0)
|
roll=0.0, thrust=0.0)
|
||||||
if key == 'a':
|
if key == 'a':
|
||||||
self.get_logger().info('left')
|
self.get_logger().info('left')
|
||||||
self.send_request(pitch=0.0, yaw=0.0,
|
self.send_request(pitch=0.0, yaw=0.0,
|
||||||
roll=-10.0, thrust=0.0)
|
roll=-1.0, thrust=0.0)
|
||||||
if key == 'd':
|
if key == 'd':
|
||||||
self.get_logger().info('right')
|
self.get_logger().info('right')
|
||||||
self.send_request(pitch=0.0, yaw=0.0,
|
self.send_request(pitch=0.0, yaw=0.0,
|
||||||
roll=10.0, thrust=0.0)
|
roll=1.0, thrust=0.0)
|
||||||
if key == 'q':
|
if key == 'q':
|
||||||
self.get_logger().info('rotate left')
|
self.get_logger().info('rotate left')
|
||||||
self.send_request(pitch=0.0, yaw=-10.0,
|
self.send_request(pitch=0.0, yaw=-1.0,
|
||||||
roll=0.0, thrust=0.0)
|
roll=0.0, thrust=0.0)
|
||||||
if key == 'e':
|
if key == 'e':
|
||||||
self.get_logger().info('rotate right')
|
self.get_logger().info('rotate right')
|
||||||
self.send_request(pitch=0.0, yaw=10.0,
|
self.send_request(pitch=0.0, yaw=1.0,
|
||||||
roll=0.0, thrust=0.0)
|
roll=0.0, thrust=0.0)
|
||||||
if key == 'z':
|
if key == 'z':
|
||||||
self.get_logger().info('down')
|
self.get_logger().info('down')
|
||||||
@@ -73,10 +73,19 @@ class TestController(Node):
|
|||||||
self.get_logger().info('up')
|
self.get_logger().info('up')
|
||||||
self.send_request(pitch=0.0, yaw=0.0,
|
self.send_request(pitch=0.0, yaw=0.0,
|
||||||
roll=0.0, thrust=0.05)
|
roll=0.0, thrust=0.05)
|
||||||
|
if key == 'v':
|
||||||
|
self.get_logger().info('down a little')
|
||||||
|
self.send_request(pitch=0.0, yaw=0.0,
|
||||||
|
roll=0.0, thrust=-0.01)
|
||||||
|
if key == 'f':
|
||||||
|
self.get_logger().info('up a little')
|
||||||
|
self.send_request(pitch=0.0, yaw=0.0,
|
||||||
|
roll=0.0, thrust=0.01)
|
||||||
if key == 'n':
|
if key == 'n':
|
||||||
self.get_logger().info('stop')
|
self.get_logger().info('stop')
|
||||||
self.send_request(pitch=0.0, yaw=0.0,
|
self.send_request(pitch=0.0, yaw=0.0,
|
||||||
roll=0.0, thrust=0.0)
|
roll=0.0, thrust=0.0)
|
||||||
|
|
||||||
# else:
|
# else:
|
||||||
# try:
|
# try:
|
||||||
# # known keys like spacebar, ctrl
|
# # known keys like spacebar, ctrl
|
||||||
|
|||||||
0
thefuck.txt
Normal file
0
thefuck.txt
Normal file
BIN
web-cam-shot.jpg
Normal file
BIN
web-cam-shot.jpg
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 1.9 MiB |
Reference in New Issue
Block a user