2020-06-24 23:16:23 +00:00
|
|
|
import math
|
2020-06-03 01:41:54 +00:00
|
|
|
import cv2
|
2020-06-01 12:58:18 +00:00
|
|
|
import logging
|
|
|
|
import time
|
|
|
|
|
2020-06-03 01:41:54 +00:00
|
|
|
import numpy as np
|
|
|
|
import pywintypes
|
2020-06-24 23:16:23 +00:00
|
|
|
import pytesseract
|
2020-06-03 01:41:54 +00:00
|
|
|
|
2020-10-15 02:23:15 +00:00
|
|
|
from fishy.engine.fullautofisher.player import Player
|
|
|
|
from fishy.engine.fullautofisher.recorder import Recorder
|
|
|
|
from fishy.engine.semifisher import fishing_event
|
|
|
|
|
2020-06-01 12:58:18 +00:00
|
|
|
from fishy.engine.IEngine import IEngine
|
2020-10-15 01:26:31 +00:00
|
|
|
from fishy.engine.fullautofisher.calibrate import callibrate
|
2020-06-03 01:41:54 +00:00
|
|
|
from fishy.engine.window import Window
|
2020-06-24 23:16:23 +00:00
|
|
|
from pynput import keyboard, mouse
|
|
|
|
|
2020-06-25 18:43:53 +00:00
|
|
|
from fishy.helper import Config, hotkey
|
2020-10-15 01:26:31 +00:00
|
|
|
from fishy.helper.helper import wait_until
|
|
|
|
from fishy.helper.hotkey import Key
|
2020-06-25 13:40:05 +00:00
|
|
|
|
2020-06-24 23:16:23 +00:00
|
|
|
mse = mouse.Controller()
|
|
|
|
kb = keyboard.Controller()
|
|
|
|
|
2020-10-15 01:26:31 +00:00
|
|
|
|
2020-06-25 13:40:05 +00:00
|
|
|
|
2020-06-25 16:39:05 +00:00
|
|
|
|
2020-06-25 13:40:05 +00:00
|
|
|
def sign(x):
|
|
|
|
return -1 if x < 0 else 1
|
2020-06-24 23:16:23 +00:00
|
|
|
|
2020-06-25 16:39:05 +00:00
|
|
|
|
2020-06-24 23:16:23 +00:00
|
|
|
def get_crop_coods(window):
|
|
|
|
Window.loop()
|
|
|
|
img = window.get_capture()
|
|
|
|
img = cv2.inRange(img, 0, 1)
|
|
|
|
Window.loop_end()
|
|
|
|
|
|
|
|
cnt, h = cv2.findContours(img, cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE)
|
|
|
|
|
|
|
|
"""
|
|
|
|
code from https://stackoverflow.com/a/45770227/4512396
|
|
|
|
"""
|
|
|
|
for i in range(len(cnt)):
|
|
|
|
area = cv2.contourArea(cnt[i])
|
|
|
|
if 5000 < area < 100000:
|
|
|
|
mask = np.zeros_like(img)
|
|
|
|
cv2.drawContours(mask, cnt, i, 255, -1)
|
|
|
|
x, y, w, h = cv2.boundingRect(cnt[i])
|
|
|
|
return x, y, x + w, y + h
|
|
|
|
|
|
|
|
|
|
|
|
def image_pre_process(img):
|
|
|
|
scale_percent = 200 # percent of original size
|
|
|
|
width = int(img.shape[1] * scale_percent / 100)
|
|
|
|
height = int(img.shape[0] * scale_percent / 100)
|
|
|
|
dim = (width, height)
|
|
|
|
img = cv2.resize(img, dim, interpolation=cv2.INTER_AREA)
|
|
|
|
img = cv2.bitwise_not(img)
|
|
|
|
return img
|
|
|
|
|
|
|
|
|
2020-06-25 01:22:39 +00:00
|
|
|
# noinspection PyBroadException
|
|
|
|
def get_values_from_image(img, tesseract_dir):
|
2020-06-24 23:16:23 +00:00
|
|
|
try:
|
2020-06-25 13:40:05 +00:00
|
|
|
pytesseract.pytesseract.tesseract_cmd = tesseract_dir + '/tesseract.exe'
|
2020-06-25 01:22:39 +00:00
|
|
|
tessdata_dir_config = f'--tessdata-dir "{tesseract_dir}" -c tessedit_char_whitelist=0123456789.'
|
2020-06-24 23:16:23 +00:00
|
|
|
|
|
|
|
text = pytesseract.image_to_string(img, lang="eng", config=tessdata_dir_config)
|
|
|
|
vals = text.split(":")
|
|
|
|
return float(vals[0]), float(vals[1]), float(vals[2])
|
2020-06-25 01:22:39 +00:00
|
|
|
except Exception:
|
2020-06-24 23:16:23 +00:00
|
|
|
logging.error("Couldn't read coods")
|
|
|
|
return None
|
2020-06-01 12:58:18 +00:00
|
|
|
|
|
|
|
|
2020-06-25 18:43:53 +00:00
|
|
|
def unassign_keys():
|
2020-10-15 01:26:31 +00:00
|
|
|
keys = [Key.UP, Key.RIGHT, Key.LEFT, Key.RIGHT]
|
2020-06-25 18:43:53 +00:00
|
|
|
for k in keys:
|
|
|
|
hotkey.free_key(k)
|
|
|
|
|
|
|
|
|
2020-06-01 12:58:18 +00:00
|
|
|
class FullAuto(IEngine):
|
2020-10-15 01:26:31 +00:00
|
|
|
rotate_by = 30
|
|
|
|
|
2020-06-24 23:16:23 +00:00
|
|
|
def __init__(self, config, gui_ref):
|
|
|
|
super().__init__(config, gui_ref)
|
2020-10-15 02:23:15 +00:00
|
|
|
self._factors = self.config.get("full_auto_factors", None)
|
|
|
|
self._tesseract_dir = None
|
|
|
|
self._target = None
|
2020-10-15 01:26:31 +00:00
|
|
|
|
2020-10-15 02:23:15 +00:00
|
|
|
if self._factors is None:
|
2020-06-25 13:40:05 +00:00
|
|
|
logging.warning("Please callibrate first")
|
2020-06-24 23:16:23 +00:00
|
|
|
|
2020-10-15 02:23:15 +00:00
|
|
|
self._hole_found_flag = False
|
|
|
|
self._curr_rotate_y = 0
|
|
|
|
|
2020-06-01 12:58:18 +00:00
|
|
|
def run(self):
|
2020-06-25 18:43:53 +00:00
|
|
|
logging.info("Loading please wait...")
|
|
|
|
self.initalize_keys()
|
2020-06-03 01:41:54 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
Window.init(False)
|
|
|
|
except pywintypes.error:
|
|
|
|
logging.info("Game window not found")
|
2020-06-25 01:22:39 +00:00
|
|
|
self.toggle_start()
|
2020-06-03 01:41:54 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
self.window = Window(color=cv2.COLOR_RGB2GRAY)
|
2020-06-24 23:16:23 +00:00
|
|
|
self.window.crop = get_crop_coods(self.window)
|
2020-10-15 02:23:15 +00:00
|
|
|
self._tesseract_dir = self.config.get("tesseract_dir", None)
|
2020-06-24 23:16:23 +00:00
|
|
|
|
2020-10-15 02:23:15 +00:00
|
|
|
if self._tesseract_dir is None:
|
2020-06-25 01:22:39 +00:00
|
|
|
logging.warning("Can't start without Tesseract Directory")
|
|
|
|
self.gui.bot_started(False)
|
|
|
|
self.toggle_start()
|
|
|
|
return
|
2020-06-03 01:41:54 +00:00
|
|
|
|
|
|
|
if self.get_gui is not None:
|
|
|
|
self.gui.bot_started(True)
|
|
|
|
|
2020-06-25 13:40:05 +00:00
|
|
|
logging.info("Controlls:\nUP: Callibrate\nLEFT: Print Coordinates\nDOWN: Set target\nRIGHT: Move to target")
|
2020-06-25 18:43:53 +00:00
|
|
|
while self.start:
|
|
|
|
Window.loop()
|
|
|
|
|
|
|
|
# self.window.show("test", func=image_pre_process)
|
|
|
|
Window.loop_end()
|
2020-06-01 12:58:18 +00:00
|
|
|
self.gui.bot_started(False)
|
2020-06-25 18:43:53 +00:00
|
|
|
unassign_keys()
|
2020-06-03 01:41:54 +00:00
|
|
|
|
2020-06-24 23:16:23 +00:00
|
|
|
def get_coods(self):
|
2020-10-15 02:23:15 +00:00
|
|
|
return get_values_from_image(self.window.processed_image(func=image_pre_process), self._tesseract_dir)
|
2020-06-24 23:16:23 +00:00
|
|
|
|
2020-06-25 13:40:05 +00:00
|
|
|
def move_to(self, target):
|
|
|
|
if target is None:
|
|
|
|
logging.error("set target first")
|
|
|
|
return
|
|
|
|
|
2020-10-15 02:23:15 +00:00
|
|
|
if self._factors is None:
|
2020-06-25 13:40:05 +00:00
|
|
|
logging.error("you need to callibrate first")
|
|
|
|
return
|
|
|
|
|
|
|
|
current = self.get_coods()
|
2020-10-15 02:23:15 +00:00
|
|
|
print(f"Moving from {(current[0], current[1])} to {self._target}")
|
2020-06-25 13:40:05 +00:00
|
|
|
move_vec = target[0] - current[0], target[1] - current[1]
|
|
|
|
target_angle = math.degrees(math.atan2(-move_vec[1], move_vec[0]))
|
2020-10-15 01:26:31 +00:00
|
|
|
from_angle = current[2]
|
|
|
|
|
|
|
|
self.rotate_to(target_angle, from_angle)
|
|
|
|
|
2020-10-15 02:23:15 +00:00
|
|
|
walking_time = math.sqrt(move_vec[0] ** 2 + move_vec[1] ** 2) / self._factors[0]
|
2020-10-15 01:26:31 +00:00
|
|
|
print(f"walking for {walking_time}")
|
|
|
|
kb.press('w')
|
|
|
|
time.sleep(walking_time)
|
|
|
|
kb.release('w')
|
|
|
|
print("done")
|
|
|
|
|
|
|
|
def rotate_to(self, target_angle, from_angle=None):
|
|
|
|
if target_angle is None:
|
|
|
|
_, _, from_angle = self.get_coods()
|
|
|
|
|
2020-06-25 13:40:05 +00:00
|
|
|
if target_angle < 0:
|
|
|
|
target_angle = 360 + target_angle
|
|
|
|
target_angle += 90
|
|
|
|
while target_angle > 360:
|
|
|
|
target_angle -= 360
|
2020-10-15 01:26:31 +00:00
|
|
|
print(f"Rotating from {from_angle} to {target_angle}")
|
2020-06-25 13:40:05 +00:00
|
|
|
|
2020-10-15 01:26:31 +00:00
|
|
|
angle_diff = target_angle - from_angle
|
2020-06-25 13:40:05 +00:00
|
|
|
|
|
|
|
if abs(angle_diff) > 180:
|
2020-06-25 16:39:05 +00:00
|
|
|
angle_diff = (360 - abs(angle_diff)) * sign(angle_diff) * -1
|
2020-06-25 13:40:05 +00:00
|
|
|
|
2020-10-15 02:23:15 +00:00
|
|
|
rotate_times = int(angle_diff / self._factors[1]) * -1
|
2020-06-25 13:40:05 +00:00
|
|
|
|
2020-10-15 01:26:31 +00:00
|
|
|
print(f"rotate_times: {rotate_times}")
|
2020-06-25 13:40:05 +00:00
|
|
|
|
|
|
|
for _ in range(abs(rotate_times)):
|
2020-10-15 01:26:31 +00:00
|
|
|
mse.move(sign(rotate_times) * FullAuto.rotate_by * -1, 0)
|
2020-06-25 13:40:05 +00:00
|
|
|
time.sleep(0.05)
|
|
|
|
|
2020-10-15 02:23:15 +00:00
|
|
|
def look_for_hole(self):
|
|
|
|
self._hole_found_flag = False
|
|
|
|
|
|
|
|
def found_hole(e):
|
|
|
|
if e == "look":
|
|
|
|
self._hole_found_flag = True
|
|
|
|
|
|
|
|
fishing_event.subscribers.append(found_hole)
|
|
|
|
|
|
|
|
t = 0
|
|
|
|
while not self._hole_found_flag or t <= self._factors[2]/2:
|
|
|
|
mse.move(0, FullAuto.rotate_by)
|
|
|
|
time.sleep(0.05)
|
|
|
|
t += 0.05
|
|
|
|
while not self._hole_found_flag or t > 0:
|
|
|
|
mse.move(0, -FullAuto.rotate_by)
|
|
|
|
time.sleep(0.05)
|
|
|
|
t -= 0.05
|
|
|
|
|
|
|
|
self._curr_rotate_y = t
|
|
|
|
fishing_event.subscribers.remove(found_hole)
|
|
|
|
return self._hole_found_flag
|
|
|
|
|
2020-06-25 18:43:53 +00:00
|
|
|
def initalize_keys(self):
|
2020-10-15 02:23:15 +00:00
|
|
|
# hotkey.set_hotkey(Key.LEFT, lambda: logging.info(self.get_coods()))
|
2020-10-15 01:26:31 +00:00
|
|
|
hotkey.set_hotkey(Key.UP, lambda: callibrate(self))
|
2020-06-25 18:43:53 +00:00
|
|
|
|
2020-10-15 02:23:15 +00:00
|
|
|
# def down():
|
|
|
|
# t = self.get_coods()[:-1]
|
|
|
|
# self.config.set("target", t)
|
|
|
|
# print(f"target_coods are {t}")
|
|
|
|
# hotkey.set_hotkey(Key.DOWN, down)
|
|
|
|
|
|
|
|
# hotkey.set_hotkey(Key.RIGHT, lambda: self.move_to(self.config.get("target", None)))
|
2020-06-25 18:43:53 +00:00
|
|
|
|
2020-10-15 02:23:15 +00:00
|
|
|
hotkey.set_hotkey(Key.LEFT, lambda: Recorder(self).start)
|
|
|
|
hotkey.set_hotkey(Key.DOWN, lambda: Player(self).start)
|
2020-06-24 23:16:23 +00:00
|
|
|
|
2020-06-03 01:41:54 +00:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
2020-06-24 23:16:23 +00:00
|
|
|
logging.getLogger("").setLevel(logging.DEBUG)
|
2020-06-25 01:22:39 +00:00
|
|
|
# noinspection PyTypeChecker
|
2020-06-25 13:40:05 +00:00
|
|
|
bot = FullAuto(Config(), None)
|
2020-06-25 18:43:53 +00:00
|
|
|
hotkey.initalize()
|
2020-06-03 01:41:54 +00:00
|
|
|
bot.toggle_start()
|