GameSentenceMiner 2.14.9__py3-none-any.whl → 2.14.10__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- GameSentenceMiner/ai/__init__.py +0 -0
- GameSentenceMiner/ai/ai_prompting.py +473 -0
- GameSentenceMiner/ocr/__init__.py +0 -0
- GameSentenceMiner/ocr/gsm_ocr_config.py +174 -0
- GameSentenceMiner/ocr/ocrconfig.py +129 -0
- GameSentenceMiner/ocr/owocr_area_selector.py +629 -0
- GameSentenceMiner/ocr/owocr_helper.py +638 -0
- GameSentenceMiner/ocr/ss_picker.py +140 -0
- GameSentenceMiner/owocr/owocr/__init__.py +1 -0
- GameSentenceMiner/owocr/owocr/__main__.py +9 -0
- GameSentenceMiner/owocr/owocr/config.py +148 -0
- GameSentenceMiner/owocr/owocr/lens_betterproto.py +1238 -0
- GameSentenceMiner/owocr/owocr/ocr.py +1690 -0
- GameSentenceMiner/owocr/owocr/run.py +1818 -0
- GameSentenceMiner/owocr/owocr/screen_coordinate_picker.py +109 -0
- GameSentenceMiner/tools/__init__.py +0 -0
- GameSentenceMiner/tools/audio_offset_selector.py +215 -0
- GameSentenceMiner/tools/ss_selector.py +135 -0
- GameSentenceMiner/tools/window_transparency.py +214 -0
- GameSentenceMiner/util/__init__.py +0 -0
- GameSentenceMiner/util/communication/__init__.py +22 -0
- GameSentenceMiner/util/communication/send.py +7 -0
- GameSentenceMiner/util/communication/websocket.py +94 -0
- GameSentenceMiner/util/configuration.py +1199 -0
- GameSentenceMiner/util/db.py +408 -0
- GameSentenceMiner/util/downloader/Untitled_json.py +472 -0
- GameSentenceMiner/util/downloader/__init__.py +0 -0
- GameSentenceMiner/util/downloader/download_tools.py +194 -0
- GameSentenceMiner/util/downloader/oneocr_dl.py +250 -0
- GameSentenceMiner/util/electron_config.py +259 -0
- GameSentenceMiner/util/ffmpeg.py +571 -0
- GameSentenceMiner/util/get_overlay_coords.py +366 -0
- GameSentenceMiner/util/gsm_utils.py +323 -0
- GameSentenceMiner/util/model.py +206 -0
- GameSentenceMiner/util/notification.py +157 -0
- GameSentenceMiner/util/text_log.py +214 -0
- GameSentenceMiner/util/win10toast/__init__.py +154 -0
- GameSentenceMiner/util/win10toast/__main__.py +22 -0
- GameSentenceMiner/web/__init__.py +0 -0
- GameSentenceMiner/web/service.py +132 -0
- GameSentenceMiner/web/static/__init__.py +0 -0
- GameSentenceMiner/web/static/apple-touch-icon.png +0 -0
- GameSentenceMiner/web/static/favicon-96x96.png +0 -0
- GameSentenceMiner/web/static/favicon.ico +0 -0
- GameSentenceMiner/web/static/favicon.svg +3 -0
- GameSentenceMiner/web/static/site.webmanifest +21 -0
- GameSentenceMiner/web/static/style.css +292 -0
- GameSentenceMiner/web/static/web-app-manifest-192x192.png +0 -0
- GameSentenceMiner/web/static/web-app-manifest-512x512.png +0 -0
- GameSentenceMiner/web/templates/__init__.py +0 -0
- GameSentenceMiner/web/templates/index.html +50 -0
- GameSentenceMiner/web/templates/text_replacements.html +238 -0
- GameSentenceMiner/web/templates/utility.html +483 -0
- GameSentenceMiner/web/texthooking_page.py +584 -0
- GameSentenceMiner/wip/__init___.py +0 -0
- {gamesentenceminer-2.14.9.dist-info → gamesentenceminer-2.14.10.dist-info}/METADATA +1 -1
- gamesentenceminer-2.14.10.dist-info/RECORD +79 -0
- gamesentenceminer-2.14.9.dist-info/RECORD +0 -24
- {gamesentenceminer-2.14.9.dist-info → gamesentenceminer-2.14.10.dist-info}/WHEEL +0 -0
- {gamesentenceminer-2.14.9.dist-info → gamesentenceminer-2.14.10.dist-info}/entry_points.txt +0 -0
- {gamesentenceminer-2.14.9.dist-info → gamesentenceminer-2.14.10.dist-info}/licenses/LICENSE +0 -0
- {gamesentenceminer-2.14.9.dist-info → gamesentenceminer-2.14.10.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,154 @@
|
|
1
|
+
from __future__ import absolute_import
|
2
|
+
from __future__ import print_function
|
3
|
+
from __future__ import unicode_literals
|
4
|
+
|
5
|
+
__all__ = ['ToastNotifier']
|
6
|
+
|
7
|
+
# #############################################################################
|
8
|
+
# ########## Libraries #############
|
9
|
+
# ##################################
|
10
|
+
# standard library
|
11
|
+
import logging
|
12
|
+
import threading
|
13
|
+
from os import path
|
14
|
+
from time import sleep
|
15
|
+
|
16
|
+
# 3rd party modules
|
17
|
+
from win32api import GetModuleHandle
|
18
|
+
from win32api import PostQuitMessage
|
19
|
+
from win32con import CW_USEDEFAULT
|
20
|
+
from win32con import IDI_APPLICATION
|
21
|
+
from win32con import IMAGE_ICON
|
22
|
+
from win32con import LR_DEFAULTSIZE
|
23
|
+
from win32con import LR_LOADFROMFILE
|
24
|
+
from win32con import WM_DESTROY
|
25
|
+
from win32con import WM_USER
|
26
|
+
from win32con import WS_OVERLAPPED
|
27
|
+
from win32con import WS_SYSMENU
|
28
|
+
from win32gui import CreateWindow
|
29
|
+
from win32gui import DestroyWindow
|
30
|
+
from win32gui import LoadIcon
|
31
|
+
from win32gui import LoadImage
|
32
|
+
from win32gui import NIF_ICON
|
33
|
+
from win32gui import NIF_INFO
|
34
|
+
from win32gui import NIF_MESSAGE
|
35
|
+
from win32gui import NIF_TIP
|
36
|
+
from win32gui import NIM_ADD
|
37
|
+
from win32gui import NIM_DELETE
|
38
|
+
from win32gui import NIM_MODIFY
|
39
|
+
from win32gui import RegisterClass
|
40
|
+
from win32gui import UnregisterClass
|
41
|
+
from win32gui import Shell_NotifyIcon
|
42
|
+
from win32gui import UpdateWindow
|
43
|
+
from win32gui import WNDCLASS
|
44
|
+
|
45
|
+
# ############################################################################
|
46
|
+
# ########### Classes ##############
|
47
|
+
# ##################################
|
48
|
+
|
49
|
+
|
50
|
+
class ToastNotifier(object):
|
51
|
+
"""Create a Windows 10 toast notification.
|
52
|
+
|
53
|
+
from: https://github.com/jithurjacob/Windows-10-Toast-Notifications
|
54
|
+
"""
|
55
|
+
|
56
|
+
def __init__(self):
|
57
|
+
"""Initialize."""
|
58
|
+
self._thread = None
|
59
|
+
|
60
|
+
def _show_toast(self, title, msg,
|
61
|
+
icon_path, duration):
|
62
|
+
"""Notification settings.
|
63
|
+
|
64
|
+
:title: notification title
|
65
|
+
:msg: notification message
|
66
|
+
:icon_path: path to the .ico file to custom notification
|
67
|
+
:duration: delay in seconds before notification self-destruction
|
68
|
+
"""
|
69
|
+
message_map = {WM_DESTROY: self.on_destroy, }
|
70
|
+
|
71
|
+
# Register the window class.
|
72
|
+
self.wc = WNDCLASS()
|
73
|
+
self.hinst = self.wc.hInstance = GetModuleHandle(None)
|
74
|
+
self.wc.lpszClassName = str("PythonTaskbar") # must be a string
|
75
|
+
self.wc.lpfnWndProc = message_map # could also specify a wndproc.
|
76
|
+
try:
|
77
|
+
self.classAtom = RegisterClass(self.wc)
|
78
|
+
except:
|
79
|
+
pass #not sure of this
|
80
|
+
style = WS_OVERLAPPED | WS_SYSMENU
|
81
|
+
self.hwnd = CreateWindow(self.classAtom, "Taskbar", style,
|
82
|
+
0, 0, CW_USEDEFAULT,
|
83
|
+
CW_USEDEFAULT,
|
84
|
+
0, 0, self.hinst, None)
|
85
|
+
UpdateWindow(self.hwnd)
|
86
|
+
|
87
|
+
# icon
|
88
|
+
if icon_path is not None:
|
89
|
+
icon_path = path.realpath(icon_path)
|
90
|
+
else:
|
91
|
+
icon_path = path.join(path.dirname(path.abspath(__file__)), 'data', 'python.ico')
|
92
|
+
icon_flags = LR_LOADFROMFILE | LR_DEFAULTSIZE
|
93
|
+
try:
|
94
|
+
hicon = LoadImage(self.hinst, icon_path,
|
95
|
+
IMAGE_ICON, 0, 0, icon_flags)
|
96
|
+
except Exception as e:
|
97
|
+
logging.error("Some trouble with the icon ({}): {}"
|
98
|
+
.format(icon_path, e))
|
99
|
+
hicon = LoadIcon(0, IDI_APPLICATION)
|
100
|
+
|
101
|
+
# Taskbar icon
|
102
|
+
flags = NIF_ICON | NIF_MESSAGE | NIF_TIP
|
103
|
+
nid = (self.hwnd, 0, flags, WM_USER + 20, hicon, "Tooltip")
|
104
|
+
Shell_NotifyIcon(NIM_ADD, nid)
|
105
|
+
Shell_NotifyIcon(NIM_MODIFY, (self.hwnd, 0, NIF_INFO,
|
106
|
+
WM_USER + 20,
|
107
|
+
hicon, "Balloon Tooltip", msg, 200,
|
108
|
+
title))
|
109
|
+
# take a rest then destroy
|
110
|
+
sleep(duration)
|
111
|
+
DestroyWindow(self.hwnd)
|
112
|
+
UnregisterClass(self.wc.lpszClassName, None)
|
113
|
+
return None
|
114
|
+
|
115
|
+
def show_toast(self, title="Notification", msg="Here comes the message",
|
116
|
+
icon_path=None, duration=5, threaded=False):
|
117
|
+
"""Notification settings.
|
118
|
+
|
119
|
+
:title: notification title
|
120
|
+
:msg: notification message
|
121
|
+
:icon_path: path to the .ico file to custom notification
|
122
|
+
:duration: delay in seconds before notification self-destruction
|
123
|
+
"""
|
124
|
+
if not threaded:
|
125
|
+
self._show_toast(title, msg, icon_path, duration)
|
126
|
+
else:
|
127
|
+
if self.notification_active():
|
128
|
+
# We have an active notification, let is finish so we don't spam them
|
129
|
+
return False
|
130
|
+
|
131
|
+
self._thread = threading.Thread(target=self._show_toast, args=(title, msg, icon_path, duration))
|
132
|
+
self._thread.start()
|
133
|
+
return True
|
134
|
+
|
135
|
+
def notification_active(self):
|
136
|
+
"""See if we have an active notification showing"""
|
137
|
+
if self._thread != None and self._thread.is_alive():
|
138
|
+
# We have an active notification, let is finish we don't spam them
|
139
|
+
return True
|
140
|
+
return False
|
141
|
+
|
142
|
+
def on_destroy(self, hwnd, msg, wparam, lparam):
|
143
|
+
"""Clean after notification ended.
|
144
|
+
|
145
|
+
:hwnd:
|
146
|
+
:msg:
|
147
|
+
:wparam:
|
148
|
+
:lparam:
|
149
|
+
"""
|
150
|
+
nid = (self.hwnd, 0)
|
151
|
+
Shell_NotifyIcon(NIM_DELETE, nid)
|
152
|
+
PostQuitMessage(0)
|
153
|
+
|
154
|
+
return 0
|
@@ -0,0 +1,22 @@
|
|
1
|
+
from win10toast import ToastNotifier
|
2
|
+
import time
|
3
|
+
|
4
|
+
# #############################################################################
|
5
|
+
# ###### Stand alone program ########
|
6
|
+
# ###################################
|
7
|
+
if __name__ == "__main__":
|
8
|
+
# Example
|
9
|
+
toaster = ToastNotifier()
|
10
|
+
toaster.show_toast(
|
11
|
+
"Hello World!!!",
|
12
|
+
"Python is 10 seconds awsm!",
|
13
|
+
duration=10)
|
14
|
+
toaster.show_toast(
|
15
|
+
"Example two",
|
16
|
+
"This notification is in it's own thread!",
|
17
|
+
icon_path=None,
|
18
|
+
duration=5,
|
19
|
+
threaded=True
|
20
|
+
)
|
21
|
+
# Wait for threaded notification to finish
|
22
|
+
while toaster.notification_active(): time.sleep(0.1)
|
File without changes
|
@@ -0,0 +1,132 @@
|
|
1
|
+
import os
|
2
|
+
import shutil
|
3
|
+
import subprocess
|
4
|
+
import threading
|
5
|
+
|
6
|
+
|
7
|
+
from GameSentenceMiner import anki
|
8
|
+
from GameSentenceMiner.util import ffmpeg, notification
|
9
|
+
from GameSentenceMiner.util.configuration import gsm_state, logger, get_config, get_temporary_directory
|
10
|
+
from GameSentenceMiner.util.ffmpeg import get_video_timings
|
11
|
+
from GameSentenceMiner.util.text_log import GameLine
|
12
|
+
|
13
|
+
|
14
|
+
def set_get_audio_from_video_callback(func):
|
15
|
+
global get_audio_from_video
|
16
|
+
get_audio_from_video = func
|
17
|
+
|
18
|
+
|
19
|
+
def handle_texthooker_button(video_path=''):
|
20
|
+
try:
|
21
|
+
if gsm_state.line_for_audio:
|
22
|
+
line: GameLine = gsm_state.line_for_audio
|
23
|
+
gsm_state.line_for_audio = None
|
24
|
+
if line == gsm_state.previous_line_for_audio:
|
25
|
+
logger.info("Line is the same as the last one, skipping processing.")
|
26
|
+
if get_config().advanced.audio_player_path:
|
27
|
+
play_audio_in_external(gsm_state.previous_audio)
|
28
|
+
elif get_config().advanced.video_player_path:
|
29
|
+
play_video_in_external(line, video_path)
|
30
|
+
else:
|
31
|
+
import sounddevice as sd
|
32
|
+
data, samplerate = gsm_state.previous_audio
|
33
|
+
sd.play(data, samplerate)
|
34
|
+
sd.wait()
|
35
|
+
return
|
36
|
+
gsm_state.previous_line_for_audio = line
|
37
|
+
if get_config().advanced.audio_player_path:
|
38
|
+
audio = get_audio_from_video(line, line.next.time if line.next else None, video_path,
|
39
|
+
temporary=True)
|
40
|
+
play_audio_in_external(audio)
|
41
|
+
gsm_state.previous_audio = audio
|
42
|
+
elif get_config().advanced.video_player_path:
|
43
|
+
play_video_in_external(line, video_path)
|
44
|
+
else:
|
45
|
+
import sounddevice as sd
|
46
|
+
import soundfile as sf
|
47
|
+
audio = get_audio_from_video(line, line.next.time if line.next else None, video_path,
|
48
|
+
temporary=True)
|
49
|
+
data, samplerate = sf.read(audio)
|
50
|
+
sd.play(data, samplerate)
|
51
|
+
sd.wait()
|
52
|
+
gsm_state.previous_audio = (data, samplerate)
|
53
|
+
return
|
54
|
+
if gsm_state.line_for_screenshot:
|
55
|
+
line: GameLine = gsm_state.line_for_screenshot
|
56
|
+
gsm_state.line_for_screenshot = None
|
57
|
+
gsm_state.previous_line_for_screenshot = line
|
58
|
+
screenshot = ffmpeg.get_screenshot_for_line(video_path, line, True)
|
59
|
+
if gsm_state.anki_note_for_screenshot:
|
60
|
+
gsm_state.anki_note_for_screenshot = None
|
61
|
+
encoded_image = ffmpeg.process_image(screenshot)
|
62
|
+
if get_config().anki.update_anki and get_config().screenshot.screenshot_hotkey_updates_anki:
|
63
|
+
last_note = anki.get_last_anki_card()
|
64
|
+
if last_note:
|
65
|
+
anki.add_image_to_card(last_note, encoded_image)
|
66
|
+
notification.send_screenshot_updated(last_note.get_field(get_config().anki.word_field))
|
67
|
+
if get_config().features.open_anki_edit:
|
68
|
+
notification.open_anki_card(last_note.noteId)
|
69
|
+
else:
|
70
|
+
notification.send_screenshot_saved(encoded_image)
|
71
|
+
else:
|
72
|
+
notification.send_screenshot_saved(encoded_image)
|
73
|
+
else:
|
74
|
+
os.startfile(screenshot)
|
75
|
+
return
|
76
|
+
except Exception as e:
|
77
|
+
logger.error(f"Error Playing Audio/Video: {e}")
|
78
|
+
logger.debug(f"Error Playing Audio/Video: {e}", exc_info=True)
|
79
|
+
return
|
80
|
+
finally:
|
81
|
+
gsm_state.previous_replay = video_path
|
82
|
+
gsm_state.videos_to_remove.add(video_path)
|
83
|
+
|
84
|
+
|
85
|
+
def play_audio_in_external(filepath):
|
86
|
+
exe = get_config().advanced.audio_player_path
|
87
|
+
|
88
|
+
filepath = os.path.normpath(filepath)
|
89
|
+
|
90
|
+
command = [exe, "--no-video", filepath]
|
91
|
+
|
92
|
+
try:
|
93
|
+
subprocess.Popen(command)
|
94
|
+
print(f"Opened {filepath} in {exe}.")
|
95
|
+
except Exception as e:
|
96
|
+
print(f"An error occurred: {e}")
|
97
|
+
|
98
|
+
|
99
|
+
def play_video_in_external(line, filepath):
|
100
|
+
command = [get_config().advanced.video_player_path]
|
101
|
+
|
102
|
+
start, _, _, _ = get_video_timings(filepath, line)
|
103
|
+
|
104
|
+
if start:
|
105
|
+
if "vlc" in get_config().advanced.video_player_path:
|
106
|
+
command.extend(["--start-time", convert_to_vlc_seconds(start), '--one-instance'])
|
107
|
+
else:
|
108
|
+
command.extend(["--start", convert_to_vlc_seconds(start)])
|
109
|
+
command.append(os.path.normpath(filepath))
|
110
|
+
|
111
|
+
logger.info(" ".join(command))
|
112
|
+
|
113
|
+
|
114
|
+
|
115
|
+
try:
|
116
|
+
subprocess.Popen(command)
|
117
|
+
logger.info(f"Opened {filepath} in {get_config().advanced.video_player_path}.")
|
118
|
+
except FileNotFoundError:
|
119
|
+
logger.error("VLC not found. Make sure it's installed and in your PATH.")
|
120
|
+
except Exception as e:
|
121
|
+
logger.error(f"An error occurred: {e}")
|
122
|
+
|
123
|
+
|
124
|
+
def convert_to_vlc_seconds(time_str):
|
125
|
+
"""Converts HH:MM:SS.milliseconds to VLC-compatible seconds."""
|
126
|
+
try:
|
127
|
+
hours, minutes, seconds_ms = time_str.split(":")
|
128
|
+
seconds, milliseconds = seconds_ms.split(".")
|
129
|
+
total_seconds = (int(hours) * 3600) + (int(minutes) * 60) + int(seconds) + (int(milliseconds) / 1000.0)
|
130
|
+
return str(total_seconds)
|
131
|
+
except ValueError:
|
132
|
+
return "Invalid time format"
|
File without changes
|
Binary file
|
Binary file
|
Binary file
|
@@ -0,0 +1,3 @@
|
|
1
|
+
<svg xmlns="http://www.w3.org/2000/svg" version="1.1" xmlns:xlink="http://www.w3.org/1999/xlink" xmlns:svgjs="http://svgjs.dev/svgjs" width="67" height="64" viewBox="0 0 67 64"><image width="67" height="64" xlink:href=""></image><style>@media (prefers-color-scheme: light) { :root { filter: none; } }
|
2
|
+
@media (prefers-color-scheme: dark) { :root { filter: none; } }
|
3
|
+
</style></svg>
|
@@ -0,0 +1,21 @@
|
|
1
|
+
{
|
2
|
+
"name": "MyWebSite",
|
3
|
+
"short_name": "MySite",
|
4
|
+
"icons": [
|
5
|
+
{
|
6
|
+
"src": "/web-app-manifest-192x192.png",
|
7
|
+
"sizes": "192x192",
|
8
|
+
"type": "image/png",
|
9
|
+
"purpose": "maskable"
|
10
|
+
},
|
11
|
+
{
|
12
|
+
"src": "/web-app-manifest-512x512.png",
|
13
|
+
"sizes": "512x512",
|
14
|
+
"type": "image/png",
|
15
|
+
"purpose": "maskable"
|
16
|
+
}
|
17
|
+
],
|
18
|
+
"theme_color": "#ffffff",
|
19
|
+
"background_color": "#ffffff",
|
20
|
+
"display": "standalone"
|
21
|
+
}
|
@@ -0,0 +1,292 @@
|
|
1
|
+
body {
|
2
|
+
background-color: #121212;
|
3
|
+
color: #e0e0e0;
|
4
|
+
font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
|
5
|
+
margin: 20px; /* Keep existing margin */
|
6
|
+
}
|
7
|
+
|
8
|
+
h1 {
|
9
|
+
color: #ffffff;
|
10
|
+
text-align: center;
|
11
|
+
font-weight: 300;
|
12
|
+
margin-bottom: 24px; /* Added margin to match previous layout */
|
13
|
+
}
|
14
|
+
|
15
|
+
/* Styles for the main container */
|
16
|
+
.container {
|
17
|
+
max-width: 64rem; /* Equivalent to max-w-4xl */
|
18
|
+
margin-left: auto;
|
19
|
+
margin-right: auto;
|
20
|
+
background-color: #1e1e1e; /* Darker background for the container */
|
21
|
+
padding: 32px; /* Equivalent to p-8 */
|
22
|
+
border-radius: 8px; /* rounded-lg */
|
23
|
+
box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1); /* shadow-md */
|
24
|
+
}
|
25
|
+
|
26
|
+
|
27
|
+
.inputField {
|
28
|
+
background-color: #1e1e1e;
|
29
|
+
color: #e0e0e0;
|
30
|
+
border: 1px solid #333;
|
31
|
+
padding: 10px;
|
32
|
+
font-size: 16px;
|
33
|
+
/* margin-bottom: 15px; Removed as it's part of flex/gap now */
|
34
|
+
border-radius: 5px;
|
35
|
+
flex-grow: 1; /* Added to make it fill space in flex container */
|
36
|
+
}
|
37
|
+
/* Placeholder color for the search input */
|
38
|
+
.inputField::placeholder {
|
39
|
+
color: #a0a0a0; /* Slightly lighter placeholder */
|
40
|
+
}
|
41
|
+
.inputField:focus {
|
42
|
+
border-color: #1a73e8; /* Highlight color on focus */
|
43
|
+
outline: none;
|
44
|
+
}
|
45
|
+
|
46
|
+
|
47
|
+
.control {
|
48
|
+
margin-bottom: 24px; /* Adjusted margin to match previous layout */
|
49
|
+
display: flex;
|
50
|
+
flex-direction: column; /* Default to column for small screens */
|
51
|
+
gap: 16px; /* Space between items */
|
52
|
+
}
|
53
|
+
|
54
|
+
@media (min-width: 640px) { /* sm breakpoint equivalent */
|
55
|
+
.control {
|
56
|
+
flex-direction: row; /* Row layout for larger screens */
|
57
|
+
gap: 16px; /* Space between items */
|
58
|
+
}
|
59
|
+
}
|
60
|
+
|
61
|
+
|
62
|
+
button {
|
63
|
+
background-color: #1a73e8;
|
64
|
+
color: #ffffff;
|
65
|
+
border: none;
|
66
|
+
padding: 10px 20px;
|
67
|
+
font-size: 16px;
|
68
|
+
cursor: pointer;
|
69
|
+
transition: background-color 0.3s;
|
70
|
+
border-radius: 5px;
|
71
|
+
}
|
72
|
+
|
73
|
+
button:disabled {
|
74
|
+
background-color: #444;
|
75
|
+
cursor: not-allowed;
|
76
|
+
}
|
77
|
+
|
78
|
+
button:hover:not(:disabled) {
|
79
|
+
background-color: #1669c1;
|
80
|
+
}
|
81
|
+
|
82
|
+
/* Specific button styles based on previous Tailwind colors */
|
83
|
+
.button-blue {
|
84
|
+
background-color: #1a73e8; /* Match existing button style */
|
85
|
+
}
|
86
|
+
.button-blue:hover:not(:disabled) {
|
87
|
+
background-color: #1669c1; /* Match existing button hover */
|
88
|
+
}
|
89
|
+
|
90
|
+
.button-green {
|
91
|
+
background-color: #34a853; /* Google Green */
|
92
|
+
}
|
93
|
+
.button-green:hover:not(:disabled) {
|
94
|
+
background-color: #2e8b4a;
|
95
|
+
}
|
96
|
+
|
97
|
+
.button-gray {
|
98
|
+
background-color: #5f6368; /* Google Gray */
|
99
|
+
}
|
100
|
+
.button-gray:hover:not(:disabled) {
|
101
|
+
background-color: #54575c;
|
102
|
+
}
|
103
|
+
|
104
|
+
|
105
|
+
select {
|
106
|
+
background-color: #1e1e1e;
|
107
|
+
color: #e0e0e0;
|
108
|
+
border: 1px solid #333;
|
109
|
+
padding: 10px;
|
110
|
+
width: 220px;
|
111
|
+
font-size: 16px;
|
112
|
+
margin-left: 5px;
|
113
|
+
margin-right: 5px;
|
114
|
+
border-radius: 5px;
|
115
|
+
}
|
116
|
+
|
117
|
+
/* Removed generic div margin to avoid conflicts */
|
118
|
+
/* div {
|
119
|
+
margin-bottom: 15px;
|
120
|
+
} */
|
121
|
+
|
122
|
+
/* Table Styles */
|
123
|
+
.table-container {
|
124
|
+
overflow-x: auto;
|
125
|
+
border-radius: 8px;
|
126
|
+
border: 1px solid #333;
|
127
|
+
}
|
128
|
+
|
129
|
+
.data-table {
|
130
|
+
width: 100%; /* Use width instead of min-width */
|
131
|
+
border-collapse: collapse;
|
132
|
+
}
|
133
|
+
|
134
|
+
.data-table thead {
|
135
|
+
background-color: #333;
|
136
|
+
}
|
137
|
+
|
138
|
+
.data-table th {
|
139
|
+
padding: 12px 24px;
|
140
|
+
text-align: left;
|
141
|
+
font-size: 0.75rem;
|
142
|
+
font-weight: 500;
|
143
|
+
color: #b0b0b0; /* Lighter gray for header text */
|
144
|
+
text-transform: uppercase;
|
145
|
+
letter-spacing: 0.05em;
|
146
|
+
}
|
147
|
+
.data-table th:last-child {
|
148
|
+
text-align: right;
|
149
|
+
}
|
150
|
+
|
151
|
+
.data-table tbody tr {
|
152
|
+
background-color: #1e1e1e; /* Match input field background */
|
153
|
+
border-bottom: 1px solid #333; /* Border between rows */
|
154
|
+
}
|
155
|
+
|
156
|
+
.data-table tbody tr:last-child {
|
157
|
+
border-bottom: none; /* No border on the last row */
|
158
|
+
}
|
159
|
+
|
160
|
+
|
161
|
+
.data-table td {
|
162
|
+
padding: 16px 24px;
|
163
|
+
white-space: nowrap;
|
164
|
+
font-size: 0.875rem;
|
165
|
+
color: #e0e0e0; /* Default text color for cells */
|
166
|
+
}
|
167
|
+
|
168
|
+
.data-table td:nth-child(1) {
|
169
|
+
font-weight: 500;
|
170
|
+
color: #ffffff; /* White color for the key */
|
171
|
+
}
|
172
|
+
|
173
|
+
.data-table td:last-child {
|
174
|
+
text-align: right;
|
175
|
+
font-weight: 500;
|
176
|
+
display: flex;
|
177
|
+
justify-content: flex-end;
|
178
|
+
gap: 8px;
|
179
|
+
}
|
180
|
+
|
181
|
+
/* Action buttons in the table */
|
182
|
+
/* Styles for Edit Button */
|
183
|
+
.action-button.edit-btn {
|
184
|
+
color: #a0c3ff; /* Lighter blue for edit */
|
185
|
+
cursor: pointer;
|
186
|
+
background: none; /* Ensure no background */
|
187
|
+
border: none; /* Ensure no border */
|
188
|
+
padding: 0; /* Remove padding */
|
189
|
+
font-size: 0.875rem;
|
190
|
+
text-decoration: underline; /* Add underline for link-like appearance */
|
191
|
+
transition: color 0.3s ease; /* Smooth transition for color */
|
192
|
+
}
|
193
|
+
.action-button.edit-btn:hover {
|
194
|
+
color: #7ba7ff; /* Darker blue on hover */
|
195
|
+
text-decoration: none; /* Remove underline on hover */
|
196
|
+
}
|
197
|
+
|
198
|
+
/* Styles for Delete Button */
|
199
|
+
.action-button.delete-btn {
|
200
|
+
color: #ff6b6b; /* Red for delete */
|
201
|
+
cursor: pointer;
|
202
|
+
background: none; /* Ensure no background */
|
203
|
+
border: none; /* Ensure no border */
|
204
|
+
padding: 0; /* Remove padding */
|
205
|
+
font-size: 0.875rem;
|
206
|
+
text-decoration: underline; /* Add underline for link-like appearance */
|
207
|
+
transition: color 0.3s ease; /* Smooth transition for color */
|
208
|
+
}
|
209
|
+
.action-button.delete-btn:hover {
|
210
|
+
color: #ff4c4c; /* Darker red on hover */
|
211
|
+
text-decoration: none; /* Remove underline on hover */
|
212
|
+
}
|
213
|
+
|
214
|
+
|
215
|
+
.no-entries-message {
|
216
|
+
color: #b0b0b0; /* Match header text color */
|
217
|
+
padding: 16px;
|
218
|
+
text-align: center;
|
219
|
+
}
|
220
|
+
|
221
|
+
/* Modal styles */
|
222
|
+
.modal {
|
223
|
+
display: none; /* Hidden by default */
|
224
|
+
position: fixed; /* Stay in place */
|
225
|
+
z-index: 1000; /* Sit on top */
|
226
|
+
left: 0;
|
227
|
+
top: 0;
|
228
|
+
width: 100%; /* Full width */
|
229
|
+
height: 100%; /* Full height */
|
230
|
+
overflow: auto; /* Enable scroll if needed */
|
231
|
+
background-color: rgba(0,0,0,0.6); /* Darker overlay */
|
232
|
+
align-items: center;
|
233
|
+
justify-content: center;
|
234
|
+
}
|
235
|
+
.modal-content {
|
236
|
+
background-color: #1e1e1e; /* Match input field background */
|
237
|
+
margin: auto;
|
238
|
+
padding: 20px;
|
239
|
+
border-radius: 8px;
|
240
|
+
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.3); /* Darker shadow */
|
241
|
+
width: 90%;
|
242
|
+
max-width: 500px;
|
243
|
+
color: #e0e0e0;
|
244
|
+
}
|
245
|
+
.close-button {
|
246
|
+
color: #b0b0b0; /* Match header text color */
|
247
|
+
float: right;
|
248
|
+
font-size: 28px;
|
249
|
+
font-weight: bold;
|
250
|
+
}
|
251
|
+
.close-button:hover,
|
252
|
+
.close-button:focus {
|
253
|
+
color: #ffffff; /* White on hover */
|
254
|
+
text-decoration: none;
|
255
|
+
cursor: pointer;
|
256
|
+
}
|
257
|
+
|
258
|
+
.form-group {
|
259
|
+
margin-bottom: 15px; /* Consistent margin */
|
260
|
+
}
|
261
|
+
|
262
|
+
.form-label {
|
263
|
+
display: block;
|
264
|
+
font-size: 0.875rem;
|
265
|
+
font-weight: 500;
|
266
|
+
color: #b0b0b0; /* Match header text color */
|
267
|
+
margin-bottom: 5px; /* Spacing below label */
|
268
|
+
}
|
269
|
+
|
270
|
+
.form-input {
|
271
|
+
display: block;
|
272
|
+
width: 95%;
|
273
|
+
padding: 10px 12px; /* Slightly more padding */
|
274
|
+
border: 1px solid #333;
|
275
|
+
border-radius: 5px; /* Match other inputs */
|
276
|
+
box-shadow: 0 1px 2px 0 rgba(0, 0, 0, 0.05);
|
277
|
+
outline: none;
|
278
|
+
background-color: #121212; /* Darker background for input */
|
279
|
+
color: #e0e0e0;
|
280
|
+
}
|
281
|
+
.form-input::placeholder {
|
282
|
+
color: #a0a0a0;
|
283
|
+
}
|
284
|
+
.form-input:focus {
|
285
|
+
border-color: #1a73e8;
|
286
|
+
box-shadow: 0 0 0 1px #1a73e8;
|
287
|
+
}
|
288
|
+
|
289
|
+
.flex-end {
|
290
|
+
display: flex;
|
291
|
+
justify-content: flex-end;
|
292
|
+
}
|
Binary file
|
Binary file
|
File without changes
|