GameSentenceMiner 2.7.17__py3-none-any.whl → 2.8.0__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/anki.py +7 -8
- GameSentenceMiner/config_gui.py +19 -3
- GameSentenceMiner/configuration.py +8 -1
- GameSentenceMiner/ffmpeg.py +1 -3
- GameSentenceMiner/gametext.py +16 -155
- GameSentenceMiner/gsm.py +28 -29
- GameSentenceMiner/obs.py +0 -3
- GameSentenceMiner/ocr/ocrconfig.py +0 -1
- GameSentenceMiner/ocr/oneocr_dl.py +243 -0
- GameSentenceMiner/ocr/owocr_area_selector.py +0 -1
- GameSentenceMiner/ocr/owocr_helper.py +25 -26
- GameSentenceMiner/text_log.py +186 -0
- GameSentenceMiner/util.py +52 -3
- GameSentenceMiner/web/__init__.py +0 -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/text_replacements.html +238 -0
- GameSentenceMiner/web/static/utility.html +313 -0
- GameSentenceMiner/web/static/web-app-manifest-192x192.png +0 -0
- GameSentenceMiner/web/static/web-app-manifest-512x512.png +0 -0
- GameSentenceMiner/web/texthooking_page.py +234 -0
- {gamesentenceminer-2.7.17.dist-info → gamesentenceminer-2.8.0.dist-info}/METADATA +2 -1
- gamesentenceminer-2.8.0.dist-info/RECORD +58 -0
- {gamesentenceminer-2.7.17.dist-info → gamesentenceminer-2.8.0.dist-info}/WHEEL +1 -1
- GameSentenceMiner/utility_gui.py +0 -204
- gamesentenceminer-2.7.17.dist-info/RECORD +0 -44
- {gamesentenceminer-2.7.17.dist-info → gamesentenceminer-2.8.0.dist-info}/entry_points.txt +0 -0
- {gamesentenceminer-2.7.17.dist-info → gamesentenceminer-2.8.0.dist-info}/licenses/LICENSE +0 -0
- {gamesentenceminer-2.7.17.dist-info → gamesentenceminer-2.8.0.dist-info}/top_level.txt +0 -0
@@ -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
|
+
}
|
@@ -0,0 +1,238 @@
|
|
1
|
+
<!DOCTYPE html>
|
2
|
+
<html lang="en">
|
3
|
+
<head>
|
4
|
+
<meta charset="UTF-8">
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
6
|
+
<title>Text Error Fixes (Electron)</title>
|
7
|
+
<link rel="stylesheet" href="/static/style.css">
|
8
|
+
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@400;600&display=swap" rel="stylesheet">
|
9
|
+
</head>
|
10
|
+
<body> <div class="container">
|
11
|
+
<h1>Text Error Fixes</h1>
|
12
|
+
|
13
|
+
<div class="control"> <button id="add-new-btn" class="button-blue">Add New Entry</button> <div class="search-container"> <input type="text" id="search-input" placeholder="Search key or value..." class="inputField"> <button id="search-button" class="button-green">Search</button> </div>
|
14
|
+
<button id="go-back-btn" class="button-gray">Go Back</button>
|
15
|
+
</div>
|
16
|
+
|
17
|
+
<div id="data-table-container" class="table-container"> <table id="data-table" class="data-table"> <thead>
|
18
|
+
<tr>
|
19
|
+
<th>
|
20
|
+
Text To Replace (Key)
|
21
|
+
</th>
|
22
|
+
<th>
|
23
|
+
Replacement Text (Value)
|
24
|
+
</th>
|
25
|
+
<th>
|
26
|
+
Actions
|
27
|
+
</th>
|
28
|
+
</tr>
|
29
|
+
</thead>
|
30
|
+
<tbody id="data-table-body">
|
31
|
+
</tbody>
|
32
|
+
</table>
|
33
|
+
<p id="no-entries-message" class="no-entries-message hidden">No entries found.</p> </div>
|
34
|
+
</div>
|
35
|
+
|
36
|
+
<div id="entry-modal" class="modal"> <div class="modal-content"> <span class="close-button">×</span> <h2 id="modal-title">Add New Entry</h2> <form id="entry-form">
|
37
|
+
<p>"re:" at the beginning = regex pattern (ex. re:.{3,}) </p>
|
38
|
+
<div class="form-group"> <label for="key-input" class="form-label">Text To Replace (Key):</label> <input type="text" id="key-input" name="key" required class="form-input"> <input type="hidden" id="original-key-input">
|
39
|
+
</div>
|
40
|
+
<div class="form-group"> <label for="value-input" class="form-label">Replacement Text (Value):</label> <input type="text" id="value-input" name="value" class="form-input">
|
41
|
+
</div>
|
42
|
+
<div class="flex-end"> <button type="submit" class="button-blue">Save Entry</button> </div>
|
43
|
+
</form>
|
44
|
+
</div>
|
45
|
+
</div>
|
46
|
+
|
47
|
+
<script>
|
48
|
+
|
49
|
+
let textData = {};
|
50
|
+
let data = {};
|
51
|
+
|
52
|
+
async function loadData() {
|
53
|
+
try {
|
54
|
+
const response = await fetch('/load-data');
|
55
|
+
if (response.ok) {
|
56
|
+
data = await response.json();
|
57
|
+
textData = data.args?.replacements || {};
|
58
|
+
} else {
|
59
|
+
console.error('Failed to load data from server');
|
60
|
+
}
|
61
|
+
} catch (error) {
|
62
|
+
console.error('Error loading data:', error);
|
63
|
+
}
|
64
|
+
renderTable();
|
65
|
+
}
|
66
|
+
|
67
|
+
async function saveData() {
|
68
|
+
try {
|
69
|
+
console.log(data)
|
70
|
+
data.args.replacements = textData;
|
71
|
+
console.log(data)
|
72
|
+
const response = await fetch('/save-data', {
|
73
|
+
method: 'POST',
|
74
|
+
headers: { 'Content-Type': 'application/json' },
|
75
|
+
body: JSON.stringify(data),
|
76
|
+
});
|
77
|
+
if (!response.ok) {
|
78
|
+
console.error('Failed to save data to server');
|
79
|
+
}
|
80
|
+
} catch (error) {
|
81
|
+
console.error('Error saving data:', error);
|
82
|
+
}
|
83
|
+
}
|
84
|
+
|
85
|
+
function renderTable(dataToRender = textData) {
|
86
|
+
const tableBody = document.getElementById('data-table-body');
|
87
|
+
let tableHtml = '';
|
88
|
+
const noEntriesMessage = document.getElementById('no-entries-message');
|
89
|
+
const dataTable = document.getElementById('data-table');
|
90
|
+
const keys = Object.keys(dataToRender);
|
91
|
+
|
92
|
+
if (keys.length === 0) {
|
93
|
+
noEntriesMessage.classList.remove('hidden');
|
94
|
+
dataTable.classList.add('hidden');
|
95
|
+
} else {
|
96
|
+
noEntriesMessage.classList.add('hidden');
|
97
|
+
dataTable.classList.remove('hidden');
|
98
|
+
keys.forEach(key => {
|
99
|
+
const value = dataToRender[key];
|
100
|
+
tableHtml += `
|
101
|
+
<tr>
|
102
|
+
<td>${escapeHTML(key)}</td> <td>${escapeHTML(value)}</td> <td>
|
103
|
+
<button class="action-button edit-btn" data-key="${escapeHTML(key)}">Edit</button>
|
104
|
+
<button class="action-button delete-button delete-btn" data-key="${escapeHTML(key)}">Delete</button>
|
105
|
+
</td>
|
106
|
+
</tr>
|
107
|
+
`;
|
108
|
+
});
|
109
|
+
tableBody.innerHTML = tableHtml;
|
110
|
+
document.querySelectorAll('.edit-btn').forEach(button => {
|
111
|
+
button.addEventListener('click', handleEditClick);
|
112
|
+
});
|
113
|
+
document.querySelectorAll('.delete-btn').forEach(button => {
|
114
|
+
button.addEventListener('click', handleDeleteClick);
|
115
|
+
});
|
116
|
+
}
|
117
|
+
}
|
118
|
+
|
119
|
+
function escapeHTML(str) {
|
120
|
+
const div = document.createElement('div');
|
121
|
+
div.appendChild(document.createTextNode(str));
|
122
|
+
return div.innerHTML;
|
123
|
+
}
|
124
|
+
|
125
|
+
const modal = document.getElementById('entry-modal');
|
126
|
+
const modalTitle = document.getElementById('modal-title');
|
127
|
+
const entryForm = document.getElementById('entry-form');
|
128
|
+
const keyInput = document.getElementById('key-input');
|
129
|
+
const valueInput = document.getElementById('value-input');
|
130
|
+
const originalKeyInput = document.getElementById('original-key-input');
|
131
|
+
const closeButton = document.querySelector('.close-button');
|
132
|
+
const addNewBtn = document.getElementById('add-new-btn');
|
133
|
+
|
134
|
+
addNewBtn.addEventListener('click', () => {
|
135
|
+
modalTitle.textContent = 'Add New Entry';
|
136
|
+
keyInput.value = '';
|
137
|
+
valueInput.value = '';
|
138
|
+
originalKeyInput.value = '';
|
139
|
+
keyInput.disabled = false;
|
140
|
+
modal.style.display = 'flex';
|
141
|
+
});
|
142
|
+
|
143
|
+
function handleEditClick(event) {
|
144
|
+
const keyToEdit = event.target.dataset.key;
|
145
|
+
const valueToEdit = textData[keyToEdit];
|
146
|
+
modalTitle.textContent = 'Edit Entry';
|
147
|
+
keyInput.value = keyToEdit;
|
148
|
+
valueInput.value = valueToEdit;
|
149
|
+
originalKeyInput.value = keyToEdit;
|
150
|
+
modal.style.display = 'flex';
|
151
|
+
}
|
152
|
+
|
153
|
+
closeButton.addEventListener('click', () => {
|
154
|
+
modal.style.display = 'none';
|
155
|
+
});
|
156
|
+
|
157
|
+
window.addEventListener('mousedown', (event) => {
|
158
|
+
if (event.target === modal) {
|
159
|
+
modal.style.display = 'none';
|
160
|
+
}
|
161
|
+
});
|
162
|
+
|
163
|
+
entryForm.addEventListener('submit', async (event) => {
|
164
|
+
event.preventDefault();
|
165
|
+
const key = keyInput.value.trim();
|
166
|
+
const value = valueInput.value.trim() || "";
|
167
|
+
const originalKey = originalKeyInput.value;
|
168
|
+
let keyEdited = false;
|
169
|
+
|
170
|
+
if (!key) {
|
171
|
+
alert('Key cannot be empty.');
|
172
|
+
return;
|
173
|
+
}
|
174
|
+
|
175
|
+
if (originalKey && originalKey !== key) {
|
176
|
+
delete textData[originalKey];
|
177
|
+
keyEdited = true;
|
178
|
+
}
|
179
|
+
|
180
|
+
if (originalKey) {
|
181
|
+
textData = { ...textData, [key]: value };
|
182
|
+
if (keyEdited) {
|
183
|
+
// No need to re-add, it was just updated
|
184
|
+
}
|
185
|
+
} else {
|
186
|
+
if (textData.hasOwnProperty(key)) {
|
187
|
+
alert(`Key "${key}" already exists. Please use the Edit function to modify it.`);
|
188
|
+
return;
|
189
|
+
}
|
190
|
+
textData = { ...textData, [key]: value };
|
191
|
+
}
|
192
|
+
|
193
|
+
await saveData();
|
194
|
+
await loadData();
|
195
|
+
modal.style.display = 'none';
|
196
|
+
});
|
197
|
+
|
198
|
+
function handleDeleteClick(event) {
|
199
|
+
const keyToDelete = event.target.dataset.key;
|
200
|
+
if (confirm(`Are you sure you want to delete the entry with key "${keyToDelete}"?`)) {
|
201
|
+
if (textData.hasOwnProperty(keyToDelete)) {
|
202
|
+
delete textData[keyToDelete];
|
203
|
+
saveData();
|
204
|
+
renderTable();
|
205
|
+
}
|
206
|
+
}
|
207
|
+
}
|
208
|
+
|
209
|
+
const searchInput = document.getElementById('search-input');
|
210
|
+
const searchButton = document.getElementById('search-button');
|
211
|
+
|
212
|
+
function performSearch() {
|
213
|
+
const query = searchInput.value.toLowerCase();
|
214
|
+
const filteredData = {};
|
215
|
+
for (const key in textData) {
|
216
|
+
if (textData.hasOwnProperty(key)) {
|
217
|
+
const value = textData[key];
|
218
|
+
if (key.toLowerCase().includes(query) || value.toLowerCase().includes(query)) {
|
219
|
+
filteredData[key] = value;
|
220
|
+
}
|
221
|
+
}
|
222
|
+
}
|
223
|
+
renderTable(filteredData);
|
224
|
+
}
|
225
|
+
|
226
|
+
searchButton.addEventListener('click', performSearch);
|
227
|
+
searchInput.addEventListener('input', performSearch);
|
228
|
+
|
229
|
+
const goBackBtn = document.getElementById('go-back-btn');
|
230
|
+
goBackBtn.addEventListener('click', () => {
|
231
|
+
window.history.back();
|
232
|
+
});
|
233
|
+
|
234
|
+
loadData();
|
235
|
+
|
236
|
+
</script>
|
237
|
+
</body>
|
238
|
+
</html>
|