Files
Kivy_APP/main.py
2025-07-25 08:08:11 +08:00

1242 lines
53 KiB
Python
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

from kivy.properties import ObjectProperty, StringProperty
from datetime import datetime, timedelta
from kivy.core.text import LabelBase
from kivy.metrics import dp
from kivy.uix.anchorlayout import AnchorLayout
from kivy.uix.boxlayout import BoxLayout
from kivy.uix.button import Button
from kivy.uix.gridlayout import GridLayout
from kivy.uix.image import Image
from kivymd.app import MDApp
from kivy.core.window import Window
from kivy.uix.screenmanager import ScreenManager
from kivy.lang import Builder
from kivy.uix.screenmanager import ScreenManager, Screen
from kivymd.uix.datatables import MDDataTable
import random
import csv
from kivymd.uix.boxlayout import MDBoxLayout
from kivymd.uix.button import MDFlatButton
from kivymd.uix.dialog import MDDialog
from kivymd.uix.label import MDLabel
from kivy.uix.label import Label
from kivy.utils import platform
from kivy.clock import Clock
from kivymd.theming import ThemeManager
LabelBase.register(name="MPoppins", fn_regular="fonts/Chinese/msyh.ttf")
LabelBase.register(name="BPoppins", fn_regular="fonts/Chinese/msyh.ttf")
LabelBase.register(name="RRubik", fn_regular="fonts/Chinese/msyh.ttf")
LabelBase.register(name="RCro", fn_regular="fonts/Chinese/msyh.ttf")
LabelBase.register(name="RPac", fn_regular="fonts/Chinese/msyh.ttf")
def get_books():
pass
class MainScreen(Screen):
pass
class LoginScreen(Screen):
login = ObjectProperty(None)
class SignUpScreen(Screen):
signup = ObjectProperty(None)
class ForgetPassScreen(Screen):
forget_Password = ObjectProperty(None)
class OtpScreen(Screen):
otp = ObjectProperty(None)
class ResetPassScreen(Screen):
reset_pass = ObjectProperty(None)
class HomeScreen(Screen):
home = ObjectProperty(None)
class ProfileScreen(Screen):
profile = ObjectProperty(None)
class ProfileEditScreen(Screen):
profile_edit = ObjectProperty(None)
class SearchBookScreen(Screen):
search_book = ObjectProperty(None)
class NotificationScreen(Screen):
notifications = ObjectProperty(None)
class RecommendScreen(Screen):
recommendation = ObjectProperty(None)
class BorrowBookScreen(Screen):
borrow_book = ObjectProperty(None)
class ReturnBookScreen(Screen):
return_book = ObjectProperty(None)
class RenewBookScreen(Screen):
renew_book = ObjectProperty(None)
class HistoryScreen(Screen):
history = ObjectProperty(None)
# Window.size = (dp(360), dp(680))
class app(MDApp):
wifi_status_text = StringProperty("")
def __init__(self):
super().__init__()
self.signup_sem = None
self.signup_branch = None
self.user_name = None
self.book_pos_name = None
self.book_pos = None
self.rec_label = None
self.rec_grid = None
self.recommend_screen = None
self.book_input = None
self.data_tables = None
self.search_grid = None
self.search_field = None
self.renew_grid = None
self.renewable_books = []
self.renew = self.renewable_books
self.history_grid = None
self.books_history = []
self.history = self.books_history
self.image_label_grid = None
self.books = []
self.items = self.books
self.book_name = None
self.isbn = None
self.a = None
self.back_button = None
self.user = None
self.dialog = None
self.profile_edit_screen = None
self.edit_email = None
self.edit_prn = None
self.edit_name = None
self.edit_number = None
self.edit_branch = None
self.profile_semester = None
self.profile_branch = None
self.profile_number = None
self.profile_prn = None
self.profile_email = None
self.profile_name = None
self.user_no = None
self.user_email = None
self.user_prn = None
self.d2 = None
self.d3 = None
self.d4 = None
self.user_pass = None
self.d1 = None
self.edit_semester = None
self.otp_button = None
self.new_pass = None
self.new_pass1 = None
self.mobile_no = None
self.password = None
self.username = None
self.theme_cls.font_styles["H1"] = ["BPoppins", 96, False, -1.5]
self.theme_cls.font_styles["H2"] = ["BPoppins", 60, False, -0.5]
self.theme_cls.font_styles["H3"] = ["BPoppins", 48, False, 0]
self.theme_cls.font_styles["H4"] = ["BPoppins", 34, False, 0.25]
self.theme_cls.font_styles["H5"] = ["BPoppins", 25, False, 0.15]
self.theme_cls.font_styles["H6"] = ["BPoppins", 16, False, 0.15]
self.theme_cls.font_styles["Subtitle1"] = ["BPoppins", 16, False, 0.15]
self.theme_cls.font_styles["Subtitle2"] = ["BPoppins", 14, False, 0.1]
self.theme_cls.font_styles["Body1"] = ["BPoppins", 16, False, 0.5]
self.theme_cls.font_styles["Body2"] = ["BPoppins", 14, False, 0.25]
self.theme_cls.font_styles["Button"] = ["BPoppins", 14, True, 1.25]
self.theme_cls.font_styles["Caption"] = ["BPoppins", 13, False, 0.15]
self.theme_cls.font_styles["Overline"] = ["BPoppins", 10, True, 1.5]
def build(self):
Builder.load_file('kv/app.kv')
screen_manager = ScreenManager()
screen_manager.add_widget(MainScreen(name="main"))
screen_manager.add_widget(HomeScreen(name="home"))
screen_manager.add_widget(LoginScreen(name="login"))
screen_manager.add_widget(SignUpScreen(name="signup"))
screen_manager.add_widget(ForgetPassScreen(name="forgot_pass"))
screen_manager.add_widget(OtpScreen(name="otp"))
screen_manager.add_widget(ResetPassScreen(name="reset_pass"))
screen_manager.add_widget(ProfileScreen(name="profile"))
screen_manager.add_widget(ProfileEditScreen(name="profile_edit"))
screen_manager.add_widget(NotificationScreen(name="notification"))
screen_manager.add_widget(SearchBookScreen(name="searchBook"))
screen_manager.add_widget(RecommendScreen(name="recommend"))
screen_manager.add_widget(BorrowBookScreen(name="borrow_book"))
screen_manager.add_widget(ReturnBookScreen(name="return_book"))
screen_manager.add_widget(RenewBookScreen(name="renew_book"))
screen_manager.add_widget(HistoryScreen(name="history"))
Clock.schedule_once(self.update_wifi_status, 2)
return screen_manager
#############################################ALL INPUT TEXT############################################################
def on_start(self):
# 从根窗口中获取名为"login"的屏幕(登录界面)
login_screen = self.root.get_screen("login")
# 通过界面ID获取登录界面中的用户名输入框组件
# self.username = login_screen.ids.username
# 通过界面ID获取登录界面中的密码输入框组件
# self.password = login_screen.ids.password
# 从根窗口中获取名为"home"的屏幕(主界面)
home_screen = self.root.get_screen("home")
# 通过界面ID获取主界面中的用户信息组件
self.user = home_screen.ids.user
# 通过界面ID获取主界面中的书架位置名称组件
self.book_pos_name = home_screen.ids.book_pos_name
# 通过界面ID获取主界面中的书架位置组件
self.book_pos = home_screen.ids.book_pos
# 从根窗口中获取名为"forgot_pass"的屏幕(忘记密码界面)
forgot_pass_screen = self.root.get_screen("forgot_pass")
# 通过界面ID获取忘记密码界面中的手机号输入框组件
self.mobile_no = forgot_pass_screen.ids.forgotPass_no
# 从根窗口中获取名为"reset_pass"的屏幕(重置密码界面)
reset_pass_screen = self.root.get_screen("reset_pass")
# 通过界面ID获取重置密码界面中的新密码输入框组件
self.new_pass = reset_pass_screen.ids.new_pass
# 通过界面ID获取重置密码界面中的确认新密码输入框组件
self.new_pass1 = reset_pass_screen.ids.new_pass1
# 从根窗口中获取名为"otp"的屏幕(验证码界面)
otp_screen = self.root.get_screen("otp")
# 通过界面ID获取验证码界面中的"验证"按钮组件
self.otp_button = otp_screen.ids.otp_button
# 通过界面ID获取验证码界面中的"返回"按钮组件
self.back_button = otp_screen.ids.back_button
# 通过界面ID获取验证码界面中的第一个数字输入框组件
self.d1 = otp_screen.ids.d1
# 通过界面ID获取验证码界面中的第二个数字输入框组件
self.d2 = otp_screen.ids.d2
# 通过界面ID获取验证码界面中的第三个数字输入框组件
self.d3 = otp_screen.ids.d3
# 通过界面ID获取验证码界面中的第四个数字输入框组件
self.d4 = otp_screen.ids.d4
# 从根窗口中获取名为"signup"的屏幕(注册界面)
signup_screen = self.root.get_screen("signup")
# 通过界面ID获取注册界面中的用户名输入框组件
self.user_name = signup_screen.ids.signup_name
# 通过界面ID获取注册界面中的PRN可能是学号/身份标识)输入框组件
self.user_prn = signup_screen.ids.signup_prn
# 通过界面ID获取注册界面中的邮箱输入框组件
self.user_email = signup_screen.ids.signup_email
# 通过界面ID获取注册界面中的手机号输入框组件
self.user_no = signup_screen.ids.signup_no
# 通过界面ID获取注册界面中的密码输入框组件
self.user_pass = signup_screen.ids.signup_pass
# 通过界面ID获取注册界面中的所属部门/专业选择组件
self.signup_branch = signup_screen.ids.signup_branch
# 通过界面ID获取注册界面中的年级/学期选择组件
self.signup_sem = signup_screen.ids.signup_sem
# 从根窗口中获取名为"profile"的屏幕(个人资料界面)
profile_screen = self.root.get_screen("profile")
# 通过界面ID获取个人资料界面中的姓名展示组件
self.profile_name = profile_screen.ids.profile_name
# 通过界面ID获取个人资料界面中的邮箱展示组件
self.profile_email = profile_screen.ids.profile_email
# 通过界面ID获取个人资料界面中的PRN身份标识展示组件
self.profile_prn = profile_screen.ids.profile_prn
# 通过界面ID获取个人资料界面中的手机号展示组件
self.profile_number = profile_screen.ids.profile_number
# 通过界面ID获取个人资料界面中的所属部门/专业展示组件
self.profile_branch = profile_screen.ids.profile_branch
# 通过界面ID获取个人资料界面中的年级/学期展示组件
self.profile_semester = profile_screen.ids.profile_semester
# 从根窗口中获取名为"profile_edit"的屏幕(个人资料编辑界面)
self.profile_edit_screen = self.root.get_screen("profile_edit")
# 通过界面ID获取资料编辑界面中的姓名编辑输入框组件
self.edit_name = self.profile_edit_screen.ids.edit_name
# 通过界面ID获取资料编辑界面中的邮箱编辑输入框组件
self.edit_email = self.profile_edit_screen.ids.edit_email
# 通过界面ID获取资料编辑界面中的PRN身份标识编辑输入框组件
self.edit_prn = self.profile_edit_screen.ids.edit_prn
# 通过界面ID获取资料编辑界面中的手机号编辑输入框组件
self.edit_number = self.profile_edit_screen.ids.edit_no
# 通过界面ID获取资料编辑界面中的所属部门/专业编辑选择组件
self.edit_branch = self.profile_edit_screen.ids.edit_branch
# 通过界面ID获取资料编辑界面中的年级/学期编辑选择组件
self.edit_semester = self.profile_edit_screen.ids.edit_sem
# 从根窗口中获取名为"borrow_book"的屏幕(借书界面)
borrow_screen = self.root.get_screen("borrow_book")
# 通过界面ID获取借书界面中的ISBN图书编号输入框组件
self.isbn = borrow_screen.ids.isbn
# 从根窗口中获取名为"return_book"的屏幕(还书界面)
return_book_screen = self.root.get_screen("return_book")
# 通过界面ID获取还书界面中的图片与标签网格布局组件
self.image_label_grid = return_book_screen.ids.image_label_grid
# 从根窗口中获取名为"renew_book"的屏幕(图书续借界面)
renew_book_screen = self.root.get_screen("renew_book")
# 通过界面ID获取续借界面中的网格布局组件
self.renew_grid = renew_book_screen.ids.renew_grid
# 从根窗口中获取名为"history"的屏幕(借阅历史界面)
history_screen = self.root.get_screen("history")
# 通过界面ID获取借阅历史界面中的网格布局组件
self.history_grid = history_screen.ids.history_grid
# 从根窗口中获取名为"searchBook"的屏幕(图书搜索界面)
search_book_screen = self.root.get_screen("searchBook")
# 通过界面ID获取图书搜索界面中的搜索输入框组件
self.search_field = search_book_screen.ids.search_field
# 通过界面ID获取图书搜索界面中的搜索结果网格布局组件
self.search_grid = search_book_screen.ids.search_grid
# 从根窗口中获取名为"recommend"的屏幕(图书推荐界面)
self.recommend_screen = self.root.get_screen("recommend")
# 通过界面ID获取推荐界面中的图书输入框组件
self.book_input = self.recommend_screen.ids.book_input
# 通过界面ID获取推荐界面中的推荐结果网格布局组件
self.rec_grid = self.recommend_screen.ids.rec_grid
# 通过界面ID获取推荐界面中的推荐信息标签组件
self.rec_label = self.recommend_screen.ids.rec_label
#############################################@Frequently used functions##################################################
@staticmethod
def change_cursor(is_enter):
"""改变鼠标指针样式"""
# 如果is_enter为True鼠标进入目标区域将鼠标指针改为手型
if is_enter:
Window.set_system_cursor('hand')
# 否则(鼠标离开目标区域),将鼠标指针改回默认箭头样式
else:
Window.set_system_cursor('arrow')
@staticmethod
def toggle_password_visibility(password_field, icon_button):
"""切换密码输入框的可见性状态"""
# 检查密码输入框当前是否处于密码隐藏状态
if password_field.password:
# 如果是隐藏状态,切换为可见状态
password_field.password = False
# 更新图标为"eye"(眼睛图标,表示当前可见)
icon_button.icon = "eye"
else:
# 如果是可见状态,切换为隐藏状态(显示为圆点/星号)
password_field.password = True
# 更新图标为"eye-off"(闭眼图标,表示当前隐藏)
icon_button.icon = "eye-off"
def clear_text(self):
"""清空界面中所有输入框和文本组件的内容"""
# 清空图书搜索框内容
self.search_field.text = ""
# 清空图书推荐输入框内容
self.book_input.text = ""
# 清空登录界面的用户名输入框内容
# self.username.text = ""
# 清空登录界面的密码输入框内容
# self.password.text = ""
# 清空忘记密码界面的手机号输入框内容
self.mobile_no.text = ""
# 清空验证码界面的第一个输入框内容
self.d1.text = ""
# 清空验证码界面的第二个输入框内容
self.d2.text = ""
# 清空验证码界面的第三个输入框内容
self.d3.text = ""
# 清空验证码界面的第四个输入框内容
self.d4.text = ""
# 清空书架位置名称文本内容
self.book_pos_name.text = ""
# 重置书架位置说明文本为示例内容
self.book_pos.text = "测试内容示例"
# 再次清空手机号输入框(可能是为了确保完全清空)
self.mobile_no.text = ""
# 清空借书界面的ISBN输入框内容
self.isbn.text = ''
def on_press_back_arrow(self):
# self.username.text = ""
# self.password.text = ""
self.mobile_no.text = ""
self.d1.text = ""
self.d2.text = ""
self.d3.text = ""
self.d4.text = ""
def focus_text_field(self, icon_button, focus, page):
if focus:
icon_button.icon_color = self.theme_cls.primary_color
elif page == "password":
icon_button.icon_color = 162 / 255, 169 / 255, 188 / 255, 1
elif page == "sign_pass":
icon_button.icon_color = 217 / 255, 217 / 255, 217 / 255, 1
else:
pass
def otp_button_fun(self, obj):
if obj == "True":
self.otp_button.text = "Reset Password"
elif obj == "1":
self.otp_button.text = "Edit Credentials"
elif obj == "2":
self.otp_button.text = "Verify OTP"
else:
self.otp_button.text = "Sign Up"
def otp_back_arrow_fun(self):
if self.otp_button.text == "Sign Up":
self.root.current = 'signup'
elif self.otp_button.text == "Reset Password":
self.root.current = 'reset_pass'
elif self.otp_button.text == "Verify OTP":
self.root.current = 'borrow_book'
############################################@DIALOG BOX most used functions##############################################
# MDDialog box function 'dialog'
def dialog1(self, text, title="Tips"):
close_button = MDFlatButton(
text="关闭",
font_name="MPoppins",
on_release=self.close_dialog
)
content_label = Label(
text=text,
font_name="MPoppins",
color=(0, 0, 0, 1),
size_hint_y=None,
height=dp(30),
halign="center",
valign="middle"
)
content_label.bind(size=content_label.setter('text_size'))
self.dialog = MDDialog(
title=title,
type="custom",
content_cls=content_label,
size_hint=(0.84, None),
buttons=[close_button]
)
self.dialog.open()
# MDDialog box dismiss function 'close_dialog'
def close_dialog(self, *args):
self.dialog.dismiss()
###################################LoginPageWork-Start#################################################
# # login Verification function "verify"
# def verify(self, obj):
# status = False
# if self.username.text == "" or self.password.text == "":
# check = "请输入账户和密码"
# return self.dialog1(check)
# else:
# name = ""
# with open('data/Users.csv', "r", encoding="utf-8") as file:
# csv_reader = csv.reader(file, delimiter=",")
# for line in csv_reader:
# try:
# if not obj:
# a = (line[1] == self.username.text or line[2] == self.username.text)
# else:
# a = (line[1] == self.edit_prn.text or line[2] == self.edit_email.text)
# if a and line[4] == self.password.text:
# status = True
# name = line[0]
# prn = line[1]
# email = line[2]
# number = line[3]
# branch = line[5]
# semester = line[6]
# break
# except IndexError:
# pass
# except Exception as e:
# print(e)
# if status:
# if not obj:
# check = "Login Successful" + "\nHello! " + name
# else:
# check = "Profile Updated Successfully!"
# self.secure_profile()
# self.root.current = "home"
# self.user.text = f"""[b]Hey! {name}[/b]"""
# self.profile_name.text = self.edit_name.text = name
# self.profile_email.text = self.edit_email.text = email
# self.profile_prn.text = self.edit_prn.text = prn
# self.profile_number.text = self.edit_number.text = number
# self.profile_semester.text = self.edit_semester.text = semester
# self.profile_branch.text = self.edit_branch.text = branch
# else:
# check = "Login Failed!. " + "Enter correct username and password."
# self.dialog1(check)
def update_wifi_status(self, dt):
self.check_wifi()
Clock.schedule_once(self.update_wifi_status, 1) # 5秒刷新一次
def check_wifi(self):
if platform != 'android':
self.wifi_status_text = '非Android设备'
return
try:
from jnius import autoclass, cast
PythonActivity = autoclass('org.kivy.android.PythonActivity')
Context = autoclass('android.content.Context')
activity = PythonActivity.mActivity
WifiManager = autoclass('android.net.wifi.WifiManager')
Formatter = autoclass('android.text.format.Formatter')
wifi_service = activity.getSystemService(Context.WIFI_SERVICE)
wifi_manager = cast('android.net.wifi.WifiManager', wifi_service)
if not wifi_manager.isWifiEnabled():
self.wifi_status_text = 'WiFi 未启用'
return # 这里直接返回不再继续取SSID等信息
wifi_info = wifi_manager.getConnectionInfo()
if wifi_info is None:
self.wifi_status_text = 'WiFi 信息不可用'
return
ssid = wifi_info.getSSID()
if ssid:
ssid_display = ssid.strip('"')
else:
ssid_display = "SSID Unknown"
ip_int = wifi_info.getIpAddress()
ip_str = Formatter.formatIpAddress(ip_int) if ip_int != 0 else "0.0.0.0"
link_speed = wifi_info.getLinkSpeed() # Mbps
bssid = wifi_info.getBSSID()
rssi = wifi_info.getRssi()
self.wifi_status_text = (
f'SSID: {ssid_display}\n'
f'IP: {ip_str}\n'
f'速度: {link_speed} Mbps\n'
f'BSSID: {bssid}\n'
f'信号: {rssi} dBm'
)
except Exception as e:
self.wifi_status_text = f'获取WiFi信息失败\n{e}'
def verify(self, obj=None):
try:
if platform == "android":
from jnius import autoclass
PythonActivity = autoclass('org.kivy.android.PythonActivity')
Context = autoclass('android.content.Context')
activity = PythonActivity.mActivity
wifi_service = activity.getSystemService(Context.WIFI_SERVICE)
wifi_info = wifi_service.getConnectionInfo()
wifi_id = wifi_info.getSSID().strip('"').lower()
else:
wifi_id = "zhizhan-2" # 非 Android 用模拟 WiFi
except Exception as e:
self.dialog1(f"获取WiFi信息失败{e}")
return
try:
with open("data/Users.csv", "r", encoding="utf-8") as file:
csv_reader = csv.reader(file)
for line in csv_reader:
# 确保至少有7列数据
if len(line) < 7:
continue
csv_ssid = line[7].strip().lower() if len(line) > 7 else ""
if wifi_id == csv_ssid or wifi_id in map(str.lower, line):
name, prn, email, number, password, branch, semester = line[:7]
self.root.current = "home"
self.user.text = f"[b]Hey! {name}[/b]"
self.profile_name.text = self.edit_name.text = name
self.profile_email.text = self.edit_email.text = email
self.profile_prn.text = self.edit_prn.text = prn
self.profile_number.text = self.edit_number.text = number
self.profile_semester.text = self.edit_semester.text = semester
self.profile_branch.text = self.edit_branch.text = branch
self.dialog1(f"欢迎你,{name} wifi认证成功")
return
except Exception as e:
self.dialog1(f"读取用户信息失败:{e}")
return
self.dialog1("当前WiFi认证失败,请检查网络或退出")
click_count = 0
def login_mode(self, username_text, button_text):
self.click_count += 1
if self.click_count % 2 == 0:
username_text.hint_text = "Enter PRN"
button_text.text = "VIT Email Login"
username_text.icon_right = "dialpad"
username_text.helper_text = "PRN of 8 digits starting with 1 2 ......only*"
username_text.max_text_length = 8
else:
username_text.hint_text = "Enter Email"
username_text.icon_right = "email-arrow-left"
button_text.text = "VIT PRN Login"
username_text.helper_text = "enter email ID of vit.edu only*"
username_text.max_text_length = 50
###############################################SIGNUP Page Functions#####################################################
# signup Verification function "check_signup"
def check_signup(self, name, prn, email, num, password, branch, sem):
if name == "" or prn == "" or email == "" or num == "" or password == "" or branch == "" or sem == "":
check = "Enter all required fields"
return self.dialog1(check)
else:
status = "None"
with open('data/Users.csv', "r") as file:
csv_reader = csv.reader(file, delimiter=',')
for line in csv_reader:
try:
if line[3] == num:
status = "num"
break
if line[1] == prn:
status = "prn"
break
if line[2] == email:
status = "email"
break
else:
status = False
except IndexError:
pass
if status == "num":
check = "Number already registered Please enter a new number"
return self.dialog1(check)
elif status == "prn":
check = "PRN already registered Please enter a PRN"
return self.dialog1(check)
elif status == "email":
check = "Email already registered Please enter a new email"
return self.dialog1(check)
else:
self.root.current = "otp"
self.send_otp(num)
check = "One Time Password has been shared on your registered Whatsapp No. Successful!."
self.dialog1(check)
##################################################FORGOT PassWord Page##################################################
# FORGOT PASSWORD page mobile number verification function 'verify_no'
def verify_no(self):
status = False
if self.mobile_no.text == "":
check = "Please enter your Mobile Number."
return self.dialog1(check)
else:
with open('data/Users.csv', "r", encoding="utf-8") as file:
csv_reader = csv.reader(file, delimiter=",")
for line in csv_reader:
try:
if line[3] == self.mobile_no.text:
status = True
break
else:
status = False
except IndexError:
pass
except Exception as e:
print(e)
if status:
check = "One Time Password has been shared on your registered Whatsapp No. Successful!."
self.root.current = "otp"
self.send_otp(self.mobile_no.text)
return self.dialog1(check)
else:
check = "Enter Registered Mobile Number."
self.dialog1(check)
def reset_password(self):
if self.new_pass.text == "" or self.new_pass1.text == "":
check = "Please enter your new password"
return self.dialog1(check)
elif not self.new_pass.text == self.new_pass1.text:
check = "Password Does Not Match. \nPlease Re-check your password."
return self.dialog1(check)
else:
with open('data/Users.csv', "r", encoding="utf-8") as file:
csv_reader = csv.DictReader(file, delimiter=",")
rows = [row for row in csv_reader]
for row in rows:
if row['Student_Whatsapp_no'] == self.mobile_no.text:
row['Student_pass'] = self.new_pass.text
break
try:
with open('data/Users.csv', "w", newline='', encoding="utf-8") as file:
headers = ["Student_Name", "Student_PRN", "Student_Email", "Student_Whatsapp_no", "Student_pass",
"Branch", "Semester"]
csv_writer = csv.DictWriter(file, fieldnames=headers)
csv_writer.writeheader()
csv_writer.writerows(rows)
check = "Password Updated Successfully!"
self.root.current = "login"
except Exception as e:
print(e)
check = "Something went wrong please try again."
self.dialog1(check)
###############################################PROFILE & EDIT Profile Page##############################################
def edit_profile(self):
if self.edit_name.text == "" or self.edit_prn.text == "" or self.edit_email.text == "" or self.edit_number.text == "":
check = "Enter all required fields"
return self.dialog1(check)
else:
with open('data/Users.csv', "r", encoding="utf-8") as file:
csv_reader = csv.DictReader(file, delimiter=",")
rows = [row for row in csv_reader]
for row in rows:
if row['Student_Whatsapp_no'] == self.profile_number.text:
row['Student_Name'] = self.edit_name.text
row['Student_PRN'] = self.edit_prn.text
row['Student_Email'] = self.edit_email.text
row['Student_Whatsapp_no'] = self.edit_number.text
row['Student_pass'] = self.password.text
row['Branch'] = self.edit_branch.text
row['Semester'] = self.edit_semester.text
break
with open('data/Users.csv', newline="", mode="w", encoding="utf-8") as file:
header = ["Student_Name", "Student_PRN", "Student_Email", "Student_Whatsapp_no", "Student_pass",
"Branch", "Semester"]
csv_writer = csv.DictWriter(file, fieldnames=header)
csv_writer.writeheader()
csv_writer.writerows(rows)
self.verify(True)
self.root.current = "home"
def secure_profile(self):
self.profile_edit_screen.ids.edit_email.readonly = True
self.profile_edit_screen.ids.edit_prn.readonly = True
self.profile_edit_screen.ids.edit_no.readonly = True
self.profile_edit_screen.ids.edit_private_credential_butn.disabled = False
#############################################OTP SEND & VERIFY PAGE#####################################################
def send_otp(self, number):
self.a = random.randint(1000, 9999)
if len(str(number)) == 10:
number1 = "+91" + str(number)
message = f"""Your verification code is {self.a}. It expires in 10 minutes. Dont share this code with anyone."""
pywhatkit.sendwhatmsg_instantly(number1, message, wait_time=15, tab_close=True, close_time=3)
elif len(str(number)) != 10:
check = "Please enter a 10 digit number."
return self.dialog1(check)
else:
pass
def verify_otp(self, d1, d2, d3, d4, otp_button):
entered_otp = str(d1) + str(d2) + str(d3) + str(d4)
if entered_otp == str(self.a):
if otp_button == "Reset Password":
self.root.current = "reset_pass"
check = "Reset your password "
return self.dialog1(check)
elif otp_button == "Edit Credentials":
self.root.current = "profile_edit"
check = "Edit your credentials "
self.profile_edit_screen.ids.edit_email.readonly = False
self.profile_edit_screen.ids.edit_prn.readonly = False
self.profile_edit_screen.ids.edit_no.readonly = False
self.profile_edit_screen.ids.edit_private_credential_butn.disabled = True
return self.dialog1(check)
elif otp_button == "Verify OTP":
self.root.current = "home"
self.add_book()
else:
data = [self.user_name.text, self.user_prn.text, self.user_email.text, self.user_no.text,
self.user_pass.text, self.signup_branch.text, self.signup_sem.text]
with open('data/Users.csv', encoding="utf-8") as file:
csv_writer = csv.writer(file, delimiter=",")
csv_writer.writerow(data)
self.root.current = "home"
check = "SignUP Successful" + "\nHello ! " + self.user_name.text
self.user.text = "H e y !" + self.user_name.text
self.verify(True)
return self.dialog1(check)
elif entered_otp != str(self.a):
check = "Please enter a correct verification code."
self.d1.text = ""
self.d2.text = ""
self.d3.text = ""
self.d4.text = ""
return self.dialog1(check)
else:
check = "Sorry! Try again later."
self.root.current = "login"
self.d1.text = ""
self.d2.text = ""
self.d3.text = ""
self.d4.text = ""
return self.dialog1(check)
#######################################################Borrow BOok######################################################
def get_book(self, isbn):
self.isbn.text = isbn
if not self.isbn.text:
check = "Please enter a valid ISBN."
return self.dialog1(check)
elif not len(self.isbn.text) == 13:
check = "Please enter a valid ISBN."
return self.dialog1(check)
else:
with open('data/Books.csv', mode="r", encoding="utf-8") as file:
csv_reader = csv.DictReader(file, delimiter=",")
rows = [row for row in csv_reader]
for row in rows:
if row["ISBN"] == self.isbn.text:
if row['Availability'] == "0":
check = "Your book is unavailable. "
return self.dialog1(check)
else:
self.otp_button_fun('2')
self.root.current = "otp"
self.send_otp(self.profile_number.text, )
def add_book(self):
with open('data/Books.csv', mode="r", encoding="utf-8") as file:
csv_reader = csv.DictReader(file, delimiter=",")
rows = [row for row in csv_reader]
for row in rows:
if row['ISBN'] == self.isbn.text:
try:
self.book_name = row['Name']
a = row['Availability']
row['Availability'] = str(int(a) - 1)
except KeyError:
pass
with open('data/Books.csv', newline="", mode="w", encoding="utf-8") as file:
headers = ["Name", "Author", "Publication Year", "Publisher", "ISBN",
"Availability"]
csv_writer = csv.DictWriter(file, fieldnames=headers)
csv_writer.writeheader()
csv_writer.writerows(rows)
self.date = datetime.now()
self.Due_date = self.date + timedelta(days=10)
details = [self.profile_prn.text, self.profile_number.text, self.book_name, self.isbn.text, self.date,
self.Due_date, ]
transaction_details = details + ["Not Returned"]
borrow_details = details + ["Not Renewed"]
files = ['data/borrowed_books.csv', 'data/all_book_transactions.csv']
for file1 in files:
with open(file1, "a", encoding="utf-8", newline='') as file:
csv_writer = csv.writer(file, delimiter=",")
if file1 == 'data/borrowed_books.csv':
csv_writer.writerow(borrow_details)
elif file1 == 'data/all_book_transactions.csv':
csv_writer.writerow(transaction_details)
check = "OTP Verified! \nBook Borrowed Successfully!"
return self.dialog1(check)
def fetch_borrowed_book(self):
with open('data/borrowed_books.csv', mode="r", encoding="utf-8") as file:
csv_reader = csv.DictReader(file, delimiter=",")
rows = [row for row in csv_reader]
self.books.clear()
for row in rows:
if row['User'] == self.profile_prn.text:
book = {'source': 'assets/sample_book_display.jpg', 'btn_text': 'Return Book', 'Book_name': row['Book_Name'],
'time': row['Due_Date'], 'Renew_status': row['Renew_status']}
self.books += [book]
print(self.books)
self.items = self.books
def fetch_renewable_books(self):
with open('data/borrowed_books.csv', mode="r", encoding="utf-8") as file:
csv_reader = csv.DictReader(file, delimiter=",")
rows = [row for row in csv_reader]
self.renewable_books.clear()
for row in rows:
a = datetime.strptime(row['Due_Date'], "%Y-%m-%d %H:%M:%S.%f")
a = a.strftime("%d-%m-%Y")
if row['User'] == self.profile_prn.text and row[
'Renew_status'] == "Not Renewed" and a == datetime.now().strftime("%d-%m-%Y"):
renew_book = {'source': 'assets/sample_book_display.jpg', 'btn_text': 'Renew Book', 'Book_name': row['Book_Name'],
'time': row['Due_Date'], 'Renew_status': row['Renew_status']}
self.renewable_books += [renew_book]
print(self.renewable_books)
self.renew = self.renewable_books
def return_book(self, obj):
if obj == 'Return':
self.fetch_borrowed_book()
self.image_label_grid.clear_widgets()
book_list = self.items
elif obj == 'Renew':
self.fetch_renewable_books()
self.renew_grid.clear_widgets()
book_list = self.renew
for item in book_list:
box = MDBoxLayout(orientation='vertical', size_hint_y=None, height="125dp")
anchor = AnchorLayout(anchor_x="center", anchor_y="center")
grid = GridLayout(cols=2, spacing=dp(20), size_hint_x=None, size_hint_y=None, size=[dp(320), dp(200)],
padding=dp(0))
img = Image(source=item['source'],
size_hint_x=None,
size_hint_y=None,
width=dp(100),
height=dp(125),
allow_stretch=True)
label = MDLabel(
text=item['Book_name'],
font_name="BPoppins",
size_hint_x=None,
size_hint_y=None,
bold=True,
size=[dp(215), dp(70)],
padding=dp(0),
on_touch_down=lambda instance, touch: self.change_cursor(True),
on_touch_up=lambda instance, touch: self.change_cursor(False),
)
date11 = item['time']
date22 = "Due Date is "
for i in range(10):
date22 += date11[i]
label1 = MDLabel(
text=date22,
font_name="BPoppins",
size_hint_x=None,
size_hint_y=None,
size=[dp(215), dp(40)],
padding=dp(0),
on_touch_down=lambda instance, touch: self.change_cursor(True),
on_touch_up=lambda instance, touch: self.change_cursor(False),
)
if item['btn_text'] == "Return Book":
btn = Button(
text=item['btn_text'],
font_name="BPoppins",
size_hint=(None, None),
size=(dp(200), dp(30)),
background_color=(0, 0, 1, 1),
on_touch_down=lambda instance, touch: self.change_cursor(True),
on_touch_up=lambda instance, touch: self.change_cursor(False),
on_release=lambda instance, src=item['source'], txt=item['btn_text'], name=item['Book_name'],
time=item['time']: self.remove_book(src, txt, name, time),
)
elif item['btn_text'] == "Renew Book":
btn = Button(
text=item['btn_text'],
font_name="BPoppins",
size_hint=(None, None),
size=(dp(200), dp(30)),
background_color=(0, 0, 1, 1),
on_touch_down=lambda instance, touch: self.change_cursor(True),
on_touch_up=lambda instance, touch: self.change_cursor(False),
on_release=lambda instance, src=item['source'], txt=item['btn_text'], name=item['Book_name'],
time=item['time']: self.renew_book(src, txt, name, time),
)
vertical_box = BoxLayout(orientation='vertical', size_hint_y=None, height=label.height + btn.height)
vertical_box.add_widget(label)
vertical_box.add_widget(label1)
vertical_box.add_widget(btn)
grid.add_widget(img)
grid.add_widget(vertical_box)
anchor.add_widget(grid)
box.add_widget(anchor)
if btn.text == "Return Book":
self.image_label_grid.add_widget(box)
elif btn.text == "Renew Book":
self.renew_grid.add_widget(box)
def remove_book(self, source, text, name, time):
with open('data/borrowed_books.csv', mode="r", encoding="utf-8") as file:
csv_reader = csv.reader(file, delimiter=",")
header = next(csv_reader)
rows = list(csv_reader)
new_rows = [row for row in rows if row[0] != self.profile_prn.text or row[2] != name or row[5] != time]
with open('data/borrowed_books.csv', mode="w", newline="", encoding="utf-8") as file:
csv_writer = csv.writer(file, delimiter=",")
csv_writer.writerow(header)
csv_writer.writerows(new_rows)
with open('data/all_book_transactions.csv', mode="r", encoding="utf-8") as file:
csv_reader = csv.DictReader(file, delimiter=",")
rows = [row for row in csv_reader]
return_date = datetime.now()
return_date = return_date.strftime("%d/%m/%Y")
for row in rows:
if row['User'] == self.profile_prn.text and row['Book_Name'] == name and row['Due_Date'] == time:
row["Return_Date"] = return_date
with open('data/all_book_transactions.csv', mode="w", newline="", encoding="utf-8") as file:
headers = ['User', 'User_No', 'Book_Name', 'ISBN', 'Date', 'Due_Date', 'Return_Date', ]
csv_writer = csv.DictWriter(file, fieldnames=headers)
csv_writer.writeheader()
csv_writer.writerows(rows)
with open('data/Books.csv', mode="r", encoding="utf-8") as file:
csv_reader = csv.DictReader(file, delimiter=",")
rows = [row for row in csv_reader]
for row in rows:
if row['Name'] == name:
try:
a = row['Availability']
row['Availability'] = str(int(a) + 1)
except KeyError:
pass
with open('data/Books.csv', newline="", mode="w", encoding="utf-8") as file:
headers = ["Name", "Author", "Publication Year", "Publisher", "ISBN",
"Availability"]
csv_writer = csv.DictWriter(file, fieldnames=headers)
csv_writer.writeheader()
csv_writer.writerows(rows)
self.items = [item for item in self.items if
not (item['source'] == source and item['btn_text'] == text and item['Book_name'] == name)]
check = "Book Returned Successfully!"
self.dialog1(check)
self.return_book("Return")
def fetch_book_history(self):
with open('data/all_book_transactions.csv', mode="r", encoding="utf-8") as file:
csv_reader = csv.DictReader(file, delimiter=",")
rows = [row for row in csv_reader]
self.books_history.clear()
for row in rows:
if row['User'] == self.profile_prn.text and row['Return_Date'] != 'Not Returned':
book = {'source': 'assets/sample_book_display.jpg', 'text': 'Rate Book', 'Book_name': row['Book_Name'],
'time': row['Date'], 'Return_Date': row['Return_Date']}
self.books_history += [book]
self.history = self.books_history
def book_history(self):
self.fetch_book_history()
self.history_grid.clear_widgets()
for item in self.history:
box = MDBoxLayout(orientation='vertical', size_hint_y=None, height="125.001dp")
anchor = AnchorLayout(anchor_x="center", anchor_y="center")
grid = GridLayout(cols=2, spacing=dp(20), size_hint_x=None, size_hint_y=None, size=[dp(320), dp(200)],
padding=dp(0))
img = Image(source=item['source'], size_hint_x=None, size_hint_y=None, width=dp(100), height=dp(125),
allow_stretch=True)
label = MDLabel(
text=item['Book_name'],
font_name="BPoppins",
size_hint_x=None,
size_hint_y=None,
bold=True,
size=[dp(215), dp(60)],
padding=dp(0),
on_touch_down=lambda instance, touch: self.change_cursor(True),
on_touch_up=lambda instance, touch: self.change_cursor(False),
)
B_date = datetime.strptime(item['time'], '%Y-%m-%d %H:%M:%S.%f')
B_date = B_date.strftime("%d-%m-%Y")
formated_B = ""
for i in range(10):
formated_B += B_date[i]
R_date = datetime.strptime(item['Return_Date'], '%d/%m/%Y')
R_date = R_date.strftime("%d-%m-%Y")
formated_R = ""
for i in range(10):
formated_R += R_date[i]
date22 = f"""Borrow Date: {formated_B}\nReturn Date: {formated_R}"""
label1 = MDLabel(
text=date22,
font_name="BPoppins",
size_hint_x=None,
size_hint_y=None,
size=[dp(215), dp(50)],
padding=dp(0),
on_touch_down=lambda instance, touch: self.change_cursor(True),
on_touch_up=lambda instance, touch: self.change_cursor(False),
)
btn = Button(
text=item['text'],
font_name="BPoppins",
size_hint=(None, None),
size=(dp(200), dp(30)),
background_color=(0, 0, 1, 1),
on_touch_down=lambda instance, touch: self.change_cursor(True),
on_touch_up=lambda instance, touch: self.change_cursor(False),
)
vertical_box = BoxLayout(orientation='vertical', size_hint_y=None, height=label.height + btn.height)
vertical_box.add_widget(label)
vertical_box.add_widget(label1)
vertical_box.add_widget(btn)
grid.add_widget(img)
grid.add_widget(vertical_box)
anchor.add_widget(grid)
box.add_widget(anchor)
self.history_grid.add_widget(box)
def data_load(self):
self.search_field.bind(text=self.search_books)
self.search_grid.clear_widgets()
self.data_tables = MDDataTable(
pos_hint={'center_x': 0.5, 'center_y': 0.4},
size_hint=(0.95, 0.5),
use_pagination=True,
padding=dp(0),
rows_num=7,
pagination_menu_pos='auto',
background_color_selected_cell=(0.8, 0.9, 1, 1),
column_data=[
("Book Name", dp(80)),
("Author", dp(50)),
("Publication Year", dp(30)),
("Publisher", dp(40)),
("ISBN", dp(30)),
("Availability", dp(30)),
],
row_data=get_books()
)
self.search_grid.add_widget(self.data_tables)
def search_books(self, instance, value):
all_data = get_books()
filtered_data = [row for row in all_data if (
value.lower() in row[0].lower() or value.lower() in row[1].lower() or value.lower() in row[4])]
self.data_tables.row_data = filtered_data
def get_recommendations(self):
book_title = self.book_input.text
recommendations = self.rec_book(book_title)
if isinstance(recommendations, str):
self.rec_label.text = recommendations
self.rec_grid.clear_widgets()
else:
self.display_rec_table(recommendations)
@staticmethod
def rec_book(book_title):
book_title_lower = book_title.lower()
if book_title_lower not in df["Name_lower"].values:
return f"{book_title} not in database"
index = df[df['Name_lower'] == book_title_lower].index[0]
distance, indices = knn.kneighbors(s_matrix[index].reshape(1, -1))
rec_title = df.iloc[indices.flatten()]["Name"].tolist()
aval = df.iloc[indices.flatten()]["Availability"].tolist()
isbn = df.iloc[indices.flatten()]["ISBN"].tolist()
recommendation = list(zip(rec_title, isbn, aval))
return recommendation
def display_rec_table(self, recommendations):
self.rec_grid.clear_widgets()
data_tables = MDDataTable(
pos_hint={'center_x': 0.5, 'center_y': 0.2},
size_hint=(0.95, 0.2),
use_pagination=True,
padding=dp(0),
rows_num=6,
pagination_menu_pos='auto',
background_color_selected_cell=(0.8, 0.9, 1, 1),
column_data=[
("Book Name", dp(80)),
("ISBN", dp(30)),
("Availability", dp(30)),
],
row_data=[(title, isbn, availability) for title, isbn, availability in recommendations]
)
self.rec_grid.add_widget(data_tables)
def where_is_book(self):
with open('data/book_positions.csv', 'r') as csvfile:
data = csv.reader(csvfile, delimiter=',')
rows = list(data)
a = self.book_pos_name.text.lower()
for row in rows:
if row[0].lower() == a:
pos = str(row[1])
self.book_pos.text = f"{a.upper()} Book is kept at {row[1]} location. Please Go and Borrow it!!\n({row[1]} means {pos[0]} th row, Left/Right side(L/R), {pos[2]} th shelf, {pos[3]} th shelf step )"
break
else:
self.book_pos.text = "Please check the Spell!"
# LabelBase.register(name="MPoppins", fn_regular="fonts/Poppins/Poppins-Medium.ttf")
# LabelBase.register(name="BPoppins", fn_regular="fonts/Poppins/Poppins-SemiBold.ttf")
# LabelBase.register(name="RRubik", fn_regular="fonts/Rubik_Vinyl/RubikVinyl-Regular.ttf")
# LabelBase.register(name="RCro", fn_regular="fonts/Croissant_One/CroissantOne-Regular.ttf")
# LabelBase.register(name="RPac", fn_regular="fonts/Pacifico/Pacifico-Regular.ttf")
if __name__ == '__main__':
app().run()