Skip to content
This repository was archived by the owner on Jun 11, 2024. It is now read-only.

Commit 1604fb3

Browse files
committed
restructure
1 parent 2078e50 commit 1604fb3

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

41 files changed

+403
-324
lines changed

autofish/bot.py

+155-246
Large diffs are not rendered by default.

autofish/gmail.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,7 @@ def verify_credentials(self):
3838
smtplib.SMTPAuthenticationError):
3939
return False
4040

41-
def send_gmail(self, to: str, subject: str = "", body: str = "", file_paths: list = None, byte_streams: list = None, screenshot: bool = False):
41+
def send(self, to: str, subject: str = "", body: str = "", file_paths: list = None, byte_streams: list = None, screenshot: bool = False) -> bool:
4242
# generating email
4343
e_mail = MIMEMultipart()
4444
e_mail["From"] = self.my_e_mail

autofish/gui.py

+8-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
import os.path
12
import tkinter as tk
23
from multiprocessing import Process
34
from tkinter.messagebox import showerror, showinfo
@@ -9,6 +10,7 @@
910
from PIL import Image as ImagePIL
1011
from PIL import ImageTk
1112

13+
from .bot import Bot
1214
from .gmail import Gmail
1315
from .load_data import resource_path, RETRIEVE_TYPES
1416

@@ -449,12 +451,17 @@ def toggle_bot(self):
449451
""" Starts/stops bot process, edits GUI accordingly"""
450452

451453
if not self.started:
454+
templates_path = resource_path(f"resources/cv_templates/{self.root.winfo_screenwidth()}x{self.root.winfo_screenheight()}")
455+
if not os.path.isdir(templates_path):
456+
showerror(title="Error!", message="Unsupported screen resolution!", parent=self.root)
457+
return
458+
452459
self.started = True
453460
self.background_image.clean_background(x=0, y=self.root.winfo_height() - 20, ind=3)
454461
self.background_image.draw_circle((10, self.root.winfo_height() - 10), 3, color="#2DFA09", filled=True)
455462
self.background_image.generate_tkinter_img()
456463
self.background_label.configure(image=self.background_image.image_tkinter)
457-
self.bot_process = Process(target=action, args=(self.retrieve, self.cast_len, self.num_of_rods, self.night_toggle, self.auto_time_warp_toggle, self.status_mails_toggle, self.email))
464+
self.bot_process = Process(target=Bot, args=((self.root.winfo_screenwidth(), self.root.winfo_screenheight()), self.retrieve, self.cast_len, self.num_of_rods, self.night_toggle, self.auto_time_warp_toggle, self.status_mails_toggle, self.email))
458465
self.bot_process.start()
459466
else:
460467
try:

autofish/load_data.py

+45-43
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,29 @@
11
import json
22
import os
33
import sys
4-
from typing import List
54

65
import cv2
76

87

9-
def get_filenames(path: str) -> List[str]:
8+
# List of possible retrieve types
9+
RETRIEVE_TYPES = [
10+
"Twitching",
11+
"Stop&Go",
12+
"Lift&Drop",
13+
"Straight",
14+
"Straight & Slow",
15+
"Popping",
16+
"Walking",
17+
"Float",
18+
"Bottom"
19+
]
20+
21+
22+
def get_filenames(path: str) -> list[str]:
1023
""" Returns list of filenames in given path """
1124

12-
filenames = [os.path.join(path, f) for f in os.listdir(path) if os.path.isfile(f)]
25+
filenames = [os.path.join(path, f) for f in os.listdir(path)]
26+
filenames = [f for f in filenames if os.path.isfile(f)]
1327
filenames.sort()
1428

1529
return filenames
@@ -23,49 +37,32 @@ def resource_path(relative_path: str) -> str:
2337
base_path = os.path.abspath(".")
2438
return os.path.join(base_path, relative_path)
2539

26-
27-
# Path to resources folder
28-
RESOURCES_PATH = resource_path("resources")
29-
30-
# List of possible retrieve types
31-
RETRIEVE_TYPES = [
32-
"Twitching",
33-
"Stop&Go",
34-
"Lift&Drop",
35-
"Straight",
36-
"Straight & Slow",
37-
"Popping",
38-
"Walking",
39-
"Float",
40-
"Bottom"
41-
]
42-
43-
4440
def load_values() -> dict:
4541
""" Loads values used by bot process """
4642

47-
with open(os.path.join(RESOURCES_PATH, "values.json"), "r") as file:
43+
with open(os.path.join(resource_path("resources"), "values.json"), "r") as file:
4844
values = json.loads(file.read())
4945

5046
return values
5147

52-
def load_templates() -> dict:
48+
def load_templates(screen_res: tuple[int, int]) -> dict:
5349
""" Loads cv templates used by bot process """
5450

51+
base_path = resource_path(f"resources/cv_templates/{screen_res[0]}x{screen_res[1]}")
5552
templates = dict()
5653

57-
templates["digits"] = load_digits()
58-
templates["fish"] = load_fish()
59-
templates["offers"] = load_offers()
60-
templates["popups"] = load_popups()
61-
templates["warp"] = load_warp()
54+
templates["digits"] = load_digits(base_path)
55+
templates["fish"] = load_fish(base_path)
56+
templates["offers"] = load_offers(base_path)
57+
templates["popups"] = load_popups(base_path)
58+
templates["warp"] = load_warp(base_path)
6259

6360
return templates
6461

65-
def load_digits() -> list:
62+
def load_digits(base_path) -> list:
6663
""" Loads digits used by bot process """
6764

68-
digits_path = os.path.join(RESOURCES_PATH, "cv_templates", "digits")
65+
digits_path = os.path.join(base_path, "digits")
6966
digits_files = get_filenames(digits_path)
7067

7168
digits = []
@@ -74,10 +71,10 @@ def load_digits() -> list:
7471

7572
return digits
7673

77-
def load_popups() -> list:
74+
def load_popups(base_path) -> list:
7875
""" Loads popups used by bot process """
7976

80-
popups_path = os.path.join(RESOURCES_PATH, "cv_templates", "popups")
77+
popups_path = os.path.join(base_path, "popups")
8178
popups_files = get_filenames(popups_path)
8279

8380
popups = []
@@ -86,43 +83,48 @@ def load_popups() -> list:
8683

8784
return popups
8885

89-
def load_offers() -> list:
86+
def load_offers(base_path) -> dict:
9087
""" Loads offers used by bot process.
9188
Returns tuple of (buy, x) templates.
9289
Buy template is used to check if there is a purchase offer.
9390
X template is used to find the location of sign X to close the offer window. """
9491

95-
offers_path = os.path.join(RESOURCES_PATH, "cv_templates", "offers")
92+
offers_path = os.path.join(base_path, "offers")
9693
offers_files = get_filenames(offers_path)
97-
9894
assert len(offers_files) == 2, "There should be exactly 2 templates for purchase offers, buy and x templates."
9995

100-
offers = []
96+
offers = dict()
10197
for offer_file in offers_files:
102-
offers.append(cv2.imread(offer_file, cv2.IMREAD_GRAYSCALE))
98+
match os.path.splitext(os.path.basename(offer_file))[0]:
99+
case "buy":
100+
offers["buy"] = cv2.imread(offer_file, cv2.IMREAD_GRAYSCALE)
101+
case "x":
102+
offers["x"] = cv2.imread(offer_file, cv2.IMREAD_GRAYSCALE)
103+
case _:
104+
raise ValueError("Invalid file name for offer template.")
103105

104106
return offers
105107

106-
def load_fish() -> dict:
108+
def load_fish(base_path) -> dict:
107109
""" Loads templates for keeping/releasing/discarding the fish """
108110

109-
fish_path = os.path.join(RESOURCES_PATH, "cv_templates", "fish")
111+
fish_path = os.path.join(base_path, "fish")
110112
fish_files = get_filenames(fish_path)
111113

112114
fish = dict()
113115
for fish_file in fish_files:
114-
fish[os.path.splitext(fish_file)[0]] = cv2.imread(fish_file, cv2.IMREAD_GRAYSCALE)
116+
fish[os.path.splitext(os.path.basename(fish_file))[0]] = cv2.imread(fish_file, cv2.IMREAD_GRAYSCALE)
115117

116118
return fish
117119

118-
def load_warp() -> dict:
120+
def load_warp(base_path) -> dict:
119121
""" Loads templates for time warp """
120122

121-
warp_path = os.path.join(RESOURCES_PATH, "cv_templates", "time_warp")
123+
warp_path = os.path.join(base_path, "warp")
122124
warp_files = get_filenames(warp_path)
123125

124126
warp = dict()
125127
for warp_file in warp_files:
126-
warp[os.path.splitext(warp_file)[0]] = cv2.imread(warp_file, cv2.IMREAD_GRAYSCALE)
128+
warp[os.path.splitext(os.path.basename(warp_file))[0]] = cv2.imread(warp_file, cv2.IMREAD_GRAYSCALE)
127129

128130
return warp

autofish/vision.py

+166-3
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,167 @@
1-
from . import load_data
1+
import time
22

3-
VALUES = load_data.load_values()
4-
TEMPLATES = load_data.load_templates()
3+
import cv2
4+
import mouse
5+
import numpy as np
6+
from PIL import ImageGrab
7+
8+
from .load_data import load_values, load_templates
9+
10+
11+
class Vision:
12+
def __init__(self, screen_res: tuple[int, int]):
13+
self.screen_res = screen_res
14+
self.values = load_values()
15+
self.cv_templates = load_templates(self.screen_res)
16+
17+
@staticmethod
18+
def color_distance(color1: tuple[int, int, int], color2: tuple[int, int, int]) -> float:
19+
""" Returns the distance between two colors in the RGB color space."""
20+
return np.linalg.norm(np.array(color1) - np.array(color2))
21+
22+
def line_length(self) -> int:
23+
""" Returns the length of the line in the game or -1 if the line length is not detected."""
24+
25+
# grab the part of the screen where the line length is displayed
26+
top_left = self.values["line_length_pos"][f"{self.screen_res[0]}x{self.screen_res[1]}"]["top_left"]
27+
bottom_right = self.values["line_length_pos"][f"{self.screen_res[0]}x{self.screen_res[1]}"]["bottom_right"]
28+
bbox = (top_left[0], top_left[1], bottom_right[0], bottom_right[1])
29+
cv_img = cv2.cvtColor(np.array(ImageGrab.grab(bbox=bbox)), cv2.COLOR_RGB2GRAY)
30+
31+
# dictionary of x positions and the detected digits and their match values at that position
32+
# x_pos_digit_map = {x_position: [(digit, match_value), ...], ...}
33+
x_pos_digit_map = dict()
34+
digits_variations = len(self.cv_templates["digits"]) // 10
35+
for i in range(len(self.cv_templates["digits"])):
36+
result = cv2.matchTemplate(cv_img, self.cv_templates["digits"][i], cv2.TM_SQDIFF)
37+
positions = np.where(result <= 11_500_000)
38+
positions_x = positions[1]
39+
positions_y = positions[0]
40+
41+
# dictionary of x positions and the minimum match value at that position for current digit
42+
x_pos_matches = dict()
43+
for x, y in zip(positions_x, positions_y):
44+
if x in x_pos_matches.keys():
45+
x_pos_matches[x] = min(x_pos_matches[x], result[y, x])
46+
else:
47+
x_pos_matches[x] = result[y, x]
48+
49+
# add the minimum value for each x position to the x_pos_digit_map
50+
for x in x_pos_matches.keys():
51+
if x in x_pos_digit_map.keys():
52+
x_pos_digit_map[x].append((i // digits_variations, x_pos_matches[x]))
53+
else:
54+
x_pos_digit_map[x] = [(i // digits_variations, x_pos_matches[x])]
55+
56+
# normalize the x positions (merge the x positions that are close to each other)
57+
coordinates = list(x_pos_digit_map.keys())
58+
norm_x_pos_digit_map = dict()
59+
while len(coordinates) != 0:
60+
coord = coordinates.pop()
61+
coord_values = x_pos_digit_map[coord]
62+
for i in range(-10, 11):
63+
if coord + i in coordinates:
64+
coord_values.extend(x_pos_digit_map[coord + i])
65+
coordinates.remove(coord + i)
66+
norm_x_pos_digit_map[coord] = coord_values
67+
68+
x_pos_ordered = list(norm_x_pos_digit_map.keys())
69+
x_pos_ordered.sort()
70+
line_length = ""
71+
for i in x_pos_ordered:
72+
digit = min(norm_x_pos_digit_map[i], key=lambda x: x[1])[0]
73+
line_length += str(digit)
74+
75+
# return the line length as an integer or -1 if the line length is not detected
76+
if line_length == "":
77+
return -1
78+
else:
79+
return int(line_length)
80+
81+
def fish_caught(self) -> bool:
82+
""" Check if the fish is caught (and process the pop-ups), return True if the fish was caught, False otherwise."""
83+
img = cv2.cvtColor(np.array(ImageGrab.grab()), cv2.COLOR_RGB2GRAY)
84+
85+
disc = cv2.minMaxLoc(cv2.matchTemplate(img, self.cv_templates["fish"]["discard"], cv2.TM_SQDIFF))
86+
if disc[0] <= 1000000:
87+
mouse.move(disc[2][0], disc[2][1], absolute=True, duration=0)
88+
mouse.click(button="left")
89+
time.sleep(2)
90+
return True
91+
92+
keep = cv2.minMaxLoc(cv2.matchTemplate(img, self.cv_templates["fish"]["keep"], cv2.TM_SQDIFF))
93+
if keep[0] <= 1000000:
94+
mouse.move(keep[2][0], keep[2][1], absolute=True, duration=0)
95+
mouse.click(button="left")
96+
time.sleep(2)
97+
return True
98+
99+
rel = cv2.minMaxLoc(cv2.matchTemplate(img, self.cv_templates["fish"]["release"], cv2.TM_SQDIFF))
100+
if rel[0] <= 1000000:
101+
mouse.move(rel[2][0], rel[2][1], absolute=True, duration=0)
102+
mouse.click(button="left")
103+
time.sleep(2)
104+
return True
105+
106+
return False
107+
108+
def fish_hooked(self) -> bool:
109+
# grab the part of the screen where the fish hooked bar is displayed
110+
top_left = self.values["fish_hooked_pos"][f"{self.screen_res[0]}x{self.screen_res[1]}"]["top_left"]
111+
bottom_right = self.values["fish_hooked_pos"][f"{self.screen_res[0]}x{self.screen_res[1]}"]["bottom_right"]
112+
bbox = (top_left[0], top_left[1], bottom_right[0], bottom_right[1])
113+
screen_load = ImageGrab.grab(bbox=bbox).load()
114+
115+
pixel = screen_load[0, 0]
116+
return self.color_distance(pixel, self.values["fish_hooked_blue"]) <= 10
117+
118+
def full_keepnet(self) -> bool:
119+
""" Check if the keepnet is full, return True if the keepnet is full, False otherwise. """
120+
121+
# grab the part of the screen where the keepnet is displayed
122+
top_left = self.values["full_keepnet_pos"][f"{self.screen_res[0]}x{self.screen_res[1]}"]["top_left"]
123+
bottom_right = self.values["full_keepnet_pos"][f"{self.screen_res[0]}x{self.screen_res[1]}"]["bottom_right"]
124+
bbox = (top_left[0], top_left[1], bottom_right[0], bottom_right[1])
125+
screen_load = ImageGrab.grab(bbox=bbox).load()
126+
127+
# check if any pixel in the keepnet is orange
128+
for i in range(0, np.shape(screen_load)[0]):
129+
if self.color_distance(screen_load[0, i], self.values["full_keepnet_orange"]) <= 10:
130+
return True
131+
return False
132+
133+
def close_popups(self) -> bool:
134+
""" Close the pop-ups that might appear on the screen. Return True if any pop-up was closed, False otherwise. """
135+
136+
ret_changes = False
137+
138+
changes = True
139+
while changes:
140+
changes = False
141+
142+
# deal with normal pop-ups (achievements, extend license, etc.)
143+
for image in self.cv_templates["popups"]:
144+
inf = cv2.minMaxLoc(cv2.matchTemplate(cv2.cvtColor(np.array(ImageGrab.grab()), cv2.COLOR_RGB2GRAY), image, cv2.TM_SQDIFF))
145+
if inf[0] <= 1000000:
146+
mouse.move(inf[2][0], inf[2][1], absolute=True, duration=0)
147+
mouse.click(button="left")
148+
changes = True
149+
ret_changes = True
150+
time.sleep(2)
151+
break
152+
if changes:
153+
continue
154+
155+
# deal with buy popup (must be closed by clicking on the "X" button)
156+
screenshot = cv2.cvtColor(np.array(ImageGrab.grab()), cv2.COLOR_RGB2GRAY)
157+
inf = cv2.minMaxLoc(cv2.matchTemplate(screenshot, self.cv_templates["offers"]["buy"], cv2.TM_SQDIFF))
158+
if inf[0] <= 1_000_000:
159+
inf2 = cv2.minMaxLoc(cv2.matchTemplate(screenshot, self.cv_templates["offers"]["x"], cv2.TM_SQDIFF))
160+
shape = np.shape(self.cv_templates["offers"]["x"])
161+
mouse.move(inf2[2][0] + shape[1] // 2, inf2[2][1] + shape[0] // 2, absolute=True, duration=0)
162+
mouse.click(button="left")
163+
changes = True
164+
ret_changes = True
165+
time.sleep(2)
166+
167+
return ret_changes

build.py

+4-2
Original file line numberDiff line numberDiff line change
@@ -62,8 +62,10 @@ def build(name, console, onefile, uac_admin, icon, files, folders):
6262
else:
6363
raise Exception("Invalid folder!")
6464

65-
PyInstaller.__main__.run(run_list)
66-
shutil.rmtree(path=work_path, ignore_errors=True)
65+
try:
66+
PyInstaller.__main__.run(run_list)
67+
finally:
68+
shutil.rmtree(path=work_path, ignore_errors=True)
6769

6870
def main():
6971
name = "Autofish"

0 commit comments

Comments
 (0)