mirror of
https://github.com/obsqrbtz/goose-highlighter.git
synced 2026-04-08 20:19:06 +03:00
fixed listmanager styling
This commit is contained in:
@@ -1,20 +1,5 @@
|
|||||||
:root {
|
@import url('../shared/colors.css');
|
||||||
--bg-color: #12100e;
|
@import url('../shared/ui-components.css');
|
||||||
--text-color: #f4ede6;
|
|
||||||
--input-bg: #1a1714;
|
|
||||||
--input-border: #3a2e26;
|
|
||||||
--button-bg: #2b211b;
|
|
||||||
--button-hover: #3a2c24;
|
|
||||||
--button-text: #f7efe9;
|
|
||||||
--accent: #f2a865;
|
|
||||||
--accent-hover: #f7c38a;
|
|
||||||
--accent-text: #1b120b;
|
|
||||||
--danger: #f87171;
|
|
||||||
--success: #4ade80;
|
|
||||||
--shadow: 0 12px 24px rgba(0, 0, 0, 0.4);
|
|
||||||
--border-radius: 14px;
|
|
||||||
--panel-bg: #17130f;
|
|
||||||
}
|
|
||||||
|
|
||||||
* {
|
* {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
@@ -23,13 +8,19 @@
|
|||||||
body {
|
body {
|
||||||
font-family: 'Inter', -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
|
font-family: 'Inter', -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
background: radial-gradient(120% 120% at 10% 0%, rgba(242, 168, 101, 0.08) 0%, transparent 45%),
|
background: radial-gradient(120% 120% at 10% 0%, rgba(204, 106, 42, 0.08) 0%, transparent 45%),
|
||||||
linear-gradient(180deg, var(--bg-color) 0%, #0f0d0b 100%);
|
linear-gradient(180deg, var(--bg-color) 0%, #f5efe9 100%);
|
||||||
color: var(--text-color);
|
color: var(--text-color);
|
||||||
min-width: 800px;
|
min-width: 800px;
|
||||||
min-height: 600px;
|
min-height: 600px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
html.dark body,
|
||||||
|
body.dark {
|
||||||
|
background: radial-gradient(120% 120% at 10% 0%, rgba(242, 168, 101, 0.08) 0%, transparent 45%),
|
||||||
|
linear-gradient(180deg, var(--bg-color) 0%, #0f0d0b 100%);
|
||||||
|
}
|
||||||
|
|
||||||
.app {
|
.app {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
@@ -41,11 +32,16 @@ body {
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
padding: 16px 20px;
|
padding: 16px 20px;
|
||||||
background: linear-gradient(145deg, #1a1511 0%, #0f0b08 100%);
|
background: linear-gradient(145deg, #f6eee7 0%, #f0e8df 100%);
|
||||||
border-bottom: 1px solid var(--input-border);
|
border-bottom: 1px solid var(--input-border);
|
||||||
box-shadow: var(--shadow);
|
box-shadow: var(--shadow);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
html.dark .topbar,
|
||||||
|
body.dark .topbar {
|
||||||
|
background: linear-gradient(145deg, #2b211b 0%, #231a14 100%);
|
||||||
|
}
|
||||||
|
|
||||||
.title {
|
.title {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@@ -155,7 +151,7 @@ body {
|
|||||||
padding-right: 4px;
|
padding-right: 4px;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Custom scrollbar styling */
|
/* Custom scrollbar styling - List Manager Specific */
|
||||||
.lists::-webkit-scrollbar,
|
.lists::-webkit-scrollbar,
|
||||||
.word-list::-webkit-scrollbar {
|
.word-list::-webkit-scrollbar {
|
||||||
width: 8px;
|
width: 8px;
|
||||||
@@ -163,7 +159,7 @@ body {
|
|||||||
|
|
||||||
.lists::-webkit-scrollbar-track,
|
.lists::-webkit-scrollbar-track,
|
||||||
.word-list::-webkit-scrollbar-track {
|
.word-list::-webkit-scrollbar-track {
|
||||||
background: #1a1511;
|
background: #f0e8df;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -176,11 +172,32 @@ body {
|
|||||||
|
|
||||||
.lists::-webkit-scrollbar-thumb:hover,
|
.lists::-webkit-scrollbar-thumb:hover,
|
||||||
.word-list::-webkit-scrollbar-thumb:hover {
|
.word-list::-webkit-scrollbar-thumb:hover {
|
||||||
|
background: #d4c4b8;
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark .lists::-webkit-scrollbar-track,
|
||||||
|
html.dark .word-list::-webkit-scrollbar-track,
|
||||||
|
body.dark .lists::-webkit-scrollbar-track,
|
||||||
|
body.dark .word-list::-webkit-scrollbar-track {
|
||||||
|
background: #1a1511;
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark .lists::-webkit-scrollbar-thumb,
|
||||||
|
html.dark .word-list::-webkit-scrollbar-thumb,
|
||||||
|
body.dark .lists::-webkit-scrollbar-thumb,
|
||||||
|
body.dark .word-list::-webkit-scrollbar-thumb {
|
||||||
|
background: var(--input-border);
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark .lists::-webkit-scrollbar-thumb:hover,
|
||||||
|
html.dark .word-list::-webkit-scrollbar-thumb:hover,
|
||||||
|
body.dark .lists::-webkit-scrollbar-thumb:hover,
|
||||||
|
body.dark .word-list::-webkit-scrollbar-thumb:hover {
|
||||||
background: #4a3e36;
|
background: #4a3e36;
|
||||||
}
|
}
|
||||||
|
|
||||||
.list-item {
|
.list-item {
|
||||||
background: #1f1813;
|
background: #ffffff;
|
||||||
border: 2px solid transparent;
|
border: 2px solid transparent;
|
||||||
border-radius: 12px;
|
border-radius: 12px;
|
||||||
padding: 10px 12px;
|
padding: 10px 12px;
|
||||||
@@ -194,23 +211,51 @@ body {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.list-item:hover {
|
.list-item:hover {
|
||||||
background: #251f19;
|
background: #f9f4f0;
|
||||||
border-color: rgba(242, 168, 101, 0.3);
|
border-color: rgba(204, 106, 42, 0.3);
|
||||||
}
|
}
|
||||||
|
|
||||||
.list-item.active {
|
.list-item.active {
|
||||||
border-color: var(--accent);
|
border-color: var(--accent);
|
||||||
box-shadow: 0 6px 12px rgba(0, 0, 0, 0.3);
|
box-shadow: 0 6px 12px rgba(0, 0, 0, 0.1);
|
||||||
background: #2a2218;
|
background: #fffaf6;
|
||||||
}
|
}
|
||||||
|
|
||||||
.list-item.selected {
|
.list-item.selected {
|
||||||
background: rgba(242, 168, 101, 0.15);
|
background: rgba(204, 106, 42, 0.1);
|
||||||
border-color: rgba(242, 168, 101, 0.6);
|
border-color: rgba(204, 106, 42, 0.5);
|
||||||
}
|
}
|
||||||
|
|
||||||
.list-item.selected.active {
|
.list-item.selected.active {
|
||||||
border-color: var(--accent);
|
border-color: var(--accent);
|
||||||
|
background: rgba(204, 106, 42, 0.15);
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark .list-item,
|
||||||
|
body.dark .list-item {
|
||||||
|
background: #1f1813;
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark .list-item:hover,
|
||||||
|
body.dark .list-item:hover {
|
||||||
|
background: #251f19;
|
||||||
|
border-color: rgba(242, 168, 101, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark .list-item.active,
|
||||||
|
body.dark .list-item.active {
|
||||||
|
box-shadow: 0 6px 12px rgba(0, 0, 0, 0.3);
|
||||||
|
background: #2a2218;
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark .list-item.selected,
|
||||||
|
body.dark .list-item.selected {
|
||||||
|
background: rgba(242, 168, 101, 0.15);
|
||||||
|
border-color: rgba(242, 168, 101, 0.6);
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark .list-item.selected.active,
|
||||||
|
body.dark .list-item.selected.active {
|
||||||
background: rgba(242, 168, 101, 0.2);
|
background: rgba(242, 168, 101, 0.2);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -322,12 +367,17 @@ body {
|
|||||||
border: 1px solid var(--input-border);
|
border: 1px solid var(--input-border);
|
||||||
border-radius: 12px;
|
border-radius: 12px;
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
background: #1a1511;
|
background: #f9f4f0;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
html.dark .word-list,
|
||||||
|
body.dark .word-list {
|
||||||
|
background: #1a1511;
|
||||||
|
}
|
||||||
|
|
||||||
.empty {
|
.empty {
|
||||||
padding: 12px;
|
padding: 12px;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
@@ -342,7 +392,7 @@ body {
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
padding: 8px 10px;
|
padding: 8px 10px;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
background: #201915;
|
background: #ffffff;
|
||||||
border: 2px solid transparent;
|
border: 2px solid transparent;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
transition: all 0.2s ease;
|
transition: all 0.2s ease;
|
||||||
@@ -350,11 +400,28 @@ body {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.word-item:hover {
|
.word-item:hover {
|
||||||
|
background: #f9f4f0;
|
||||||
|
border-color: rgba(204, 106, 42, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.word-item.selected {
|
||||||
|
background: rgba(204, 106, 42, 0.1);
|
||||||
|
border-color: rgba(204, 106, 42, 0.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark .word-item,
|
||||||
|
body.dark .word-item {
|
||||||
|
background: #201915;
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark .word-item:hover,
|
||||||
|
body.dark .word-item:hover {
|
||||||
background: #2a2218;
|
background: #2a2218;
|
||||||
border-color: rgba(242, 168, 101, 0.2);
|
border-color: rgba(242, 168, 101, 0.2);
|
||||||
}
|
}
|
||||||
|
|
||||||
.word-item.selected {
|
html.dark .word-item.selected,
|
||||||
|
body.dark .word-item.selected {
|
||||||
background: rgba(242, 168, 101, 0.15);
|
background: rgba(242, 168, 101, 0.15);
|
||||||
border-color: rgba(242, 168, 101, 0.6);
|
border-color: rgba(242, 168, 101, 0.6);
|
||||||
}
|
}
|
||||||
@@ -401,111 +468,7 @@ body {
|
|||||||
pointer-events: auto;
|
pointer-events: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
.icon-btn {
|
|
||||||
background: transparent;
|
|
||||||
border: none;
|
|
||||||
color: var(--text-color);
|
|
||||||
opacity: 0.6;
|
|
||||||
cursor: pointer;
|
|
||||||
padding: 4px 6px;
|
|
||||||
border-radius: 6px;
|
|
||||||
transition: all 0.2s ease;
|
|
||||||
font-size: 0.85rem;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
min-width: 28px;
|
|
||||||
min-height: 28px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.icon-btn:hover {
|
|
||||||
opacity: 1;
|
|
||||||
background: rgba(242, 168, 101, 0.15);
|
|
||||||
color: var(--accent);
|
|
||||||
}
|
|
||||||
|
|
||||||
.icon-btn i {
|
|
||||||
pointer-events: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.toggle-btn {
|
|
||||||
width: 40px;
|
|
||||||
height: 22px;
|
|
||||||
background: rgba(255, 255, 255, 0.1);
|
|
||||||
border: 1px solid var(--input-border);
|
|
||||||
border-radius: 11px;
|
|
||||||
position: relative;
|
|
||||||
cursor: pointer;
|
|
||||||
transition: all 0.2s ease;
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.toggle-btn::after {
|
|
||||||
content: '';
|
|
||||||
position: absolute;
|
|
||||||
width: 16px;
|
|
||||||
height: 16px;
|
|
||||||
background: var(--text-color);
|
|
||||||
border-radius: 50%;
|
|
||||||
top: 2px;
|
|
||||||
left: 2px;
|
|
||||||
transition: all 0.2s ease;
|
|
||||||
}
|
|
||||||
|
|
||||||
.toggle-btn.active {
|
|
||||||
background: var(--accent);
|
|
||||||
border-color: var(--accent);
|
|
||||||
}
|
|
||||||
|
|
||||||
.toggle-btn.active::after {
|
|
||||||
left: 20px;
|
|
||||||
background: var(--accent-text);
|
|
||||||
}
|
|
||||||
|
|
||||||
button {
|
|
||||||
border: none;
|
|
||||||
border-radius: 10px;
|
|
||||||
padding: 8px 10px;
|
|
||||||
background: var(--button-bg);
|
|
||||||
color: var(--button-text);
|
|
||||||
cursor: pointer;
|
|
||||||
transition: background 0.2s ease;
|
|
||||||
font-size: 0.85rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
button:hover {
|
|
||||||
background: var(--button-hover);
|
|
||||||
}
|
|
||||||
|
|
||||||
button.primary {
|
|
||||||
background: var(--accent);
|
|
||||||
color: var(--accent-text);
|
|
||||||
}
|
|
||||||
|
|
||||||
button.primary:hover {
|
|
||||||
background: var(--accent-hover);
|
|
||||||
}
|
|
||||||
|
|
||||||
button.danger {
|
|
||||||
background: rgba(248, 113, 113, 0.15);
|
|
||||||
color: var(--danger);
|
|
||||||
border: 1px solid rgba(248, 113, 113, 0.4);
|
|
||||||
}
|
|
||||||
|
|
||||||
button.ghost {
|
|
||||||
background: transparent;
|
|
||||||
border: 1px solid var(--input-border);
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="text"],
|
|
||||||
select,
|
|
||||||
input[type="color"] {
|
|
||||||
background: var(--input-bg);
|
|
||||||
color: var(--text-color);
|
|
||||||
border: 1px solid var(--input-border);
|
|
||||||
border-radius: 10px;
|
|
||||||
padding: 6px 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="color"] {
|
input[type="color"] {
|
||||||
padding: 0;
|
padding: 0;
|
||||||
@@ -516,16 +479,6 @@ input[type="color"] {
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
input[type="color"]::-webkit-color-swatch-wrapper {
|
|
||||||
padding: 6px;
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="color"]::-webkit-color-swatch {
|
|
||||||
border: none;
|
|
||||||
border-radius: 8px;
|
|
||||||
box-shadow: inset 0 0 0 1px rgba(255, 255, 255, 0.08);
|
|
||||||
}
|
|
||||||
|
|
||||||
.colors input[type="color"] {
|
.colors input[type="color"] {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
@@ -537,14 +490,6 @@ input[type="color"]::-webkit-color-swatch {
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
.word-item input[type="color"]::-webkit-color-swatch-wrapper {
|
|
||||||
padding: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.word-item input[type="color"]::-webkit-color-swatch {
|
|
||||||
border-radius: 6px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.word-item input[type="color"]:hover {
|
.word-item input[type="color"]:hover {
|
||||||
transform: scale(1.05);
|
transform: scale(1.05);
|
||||||
transition: transform 0.2s ease;
|
transition: transform 0.2s ease;
|
||||||
@@ -555,6 +500,102 @@ input[type="color"]::-webkit-color-swatch {
|
|||||||
opacity: 0.7;
|
opacity: 0.7;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.pagination-container {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 12px;
|
||||||
|
padding: 12px;
|
||||||
|
background: #f9f4f0;
|
||||||
|
border: 1px solid var(--input-border);
|
||||||
|
border-radius: 12px;
|
||||||
|
margin-top: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark .pagination-container,
|
||||||
|
body.dark .pagination-container {
|
||||||
|
background: #1a1511;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagination-info {
|
||||||
|
font-size: 0.85rem;
|
||||||
|
opacity: 0.8;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagination-controls {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagination-btn {
|
||||||
|
background: var(--button-bg);
|
||||||
|
border: 1px solid var(--input-border);
|
||||||
|
color: var(--button-text);
|
||||||
|
border-radius: 8px;
|
||||||
|
padding: 6px 8px;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
min-width: 32px;
|
||||||
|
min-height: 32px;
|
||||||
|
font-size: 0.8rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagination-btn:hover:not(:disabled) {
|
||||||
|
background: var(--button-hover);
|
||||||
|
border-color: var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagination-btn:disabled {
|
||||||
|
opacity: 0.4;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagination-pages {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-info {
|
||||||
|
font-size: 0.85rem;
|
||||||
|
opacity: 0.9;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-size-controls {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 8px;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-size-controls label {
|
||||||
|
opacity: 0.8;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-size-select {
|
||||||
|
background: var(--input-bg);
|
||||||
|
color: var(--text-color);
|
||||||
|
border: 1px solid var(--input-border);
|
||||||
|
border-radius: 8px;
|
||||||
|
padding: 4px 8px;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
min-width: 60px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-size-select:focus {
|
||||||
|
outline: none;
|
||||||
|
border-color: var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@media (max-width: 920px) {
|
@media (max-width: 920px) {
|
||||||
body {
|
body {
|
||||||
min-width: 100%;
|
min-width: 100%;
|
||||||
@@ -564,4 +605,22 @@ input[type="color"]::-webkit-color-swatch {
|
|||||||
grid-template-columns: 1fr;
|
grid-template-columns: 1fr;
|
||||||
height: auto;
|
height: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.pagination-container {
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagination-controls {
|
||||||
|
order: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagination-info {
|
||||||
|
order: 1;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-size-controls {
|
||||||
|
order: 3;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,7 +20,11 @@
|
|||||||
<div class="subtitle">List Manager</div>
|
<div class="subtitle">List Manager</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="topbar-actions">
|
<div class="topbar-actions">
|
||||||
|
<label class="icon-toggle" title="Toggle dark mode">
|
||||||
|
<input type="checkbox" class="hidden-toggle" id="themeToggle" />
|
||||||
|
<i class="toggle-icon theme-icon fa-solid"></i>
|
||||||
|
</label>
|
||||||
<button id="exportListBtn" class="ghost"><i class="fa-solid fa-download"></i> Export List</button>
|
<button id="exportListBtn" class="ghost"><i class="fa-solid fa-download"></i> Export List</button>
|
||||||
<button id="newListBtn" class="primary"><i class="fa-solid fa-plus"></i> New List</button>
|
<button id="newListBtn" class="primary"><i class="fa-solid fa-plus"></i> New List</button>
|
||||||
</div>
|
</div>
|
||||||
@@ -88,7 +92,8 @@
|
|||||||
<div class="selection-hint">Click to select • Ctrl/Cmd+Click for multi-select • Click edit icon to rename</div>
|
<div class="selection-hint">Click to select • Ctrl/Cmd+Click for multi-select • Click edit icon to rename</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="wordList" class="word-list"></div>
|
<div id="wordList" class="word-list"></div>
|
||||||
|
<div id="paginationControls" class="pagination-container"></div>
|
||||||
</section>
|
</section>
|
||||||
</main>
|
</main>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
1
package-lock.json
generated
1
package-lock.json
generated
@@ -4,7 +4,6 @@
|
|||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "goose-highlighter",
|
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@eslint/css": "^0.9.0",
|
"@eslint/css": "^0.9.0",
|
||||||
"@eslint/js": "^9.30.0",
|
"@eslint/js": "^9.30.0",
|
||||||
|
|||||||
317
popup/popup.css
317
popup/popup.css
@@ -1,32 +1,10 @@
|
|||||||
:root {
|
@import url('../shared/colors.css');
|
||||||
--bg-color: #12100e;
|
@import url('../shared/ui-components.css');
|
||||||
--text-color: #f4ede6;
|
|
||||||
--input-bg: #1a1714;
|
|
||||||
--input-border: #3a2e26;
|
|
||||||
--button-bg: #2b211b;
|
|
||||||
--button-hover: #3a2c24;
|
|
||||||
--button-text: #f7efe9;
|
|
||||||
--accent: #f2a865;
|
|
||||||
--accent-hover: #f7c38a;
|
|
||||||
--accent-text: #1b120b;
|
|
||||||
--highlight-tag: #231a14;
|
|
||||||
--highlight-tag-border: #46372c;
|
|
||||||
--danger: #f87171;
|
|
||||||
--success: #4ade80;
|
|
||||||
--shadow: 0 10px 22px rgba(0, 0, 0, 0.5);
|
|
||||||
--shadow-sm: 0 4px 10px rgba(0, 0, 0, 0.4);
|
|
||||||
--border-radius: 12px;
|
|
||||||
--section-bg: #17130f;
|
|
||||||
--switch-bg: #4b3a2f;
|
|
||||||
--checkbox-accent: #f2a865;
|
|
||||||
--checkbox-border: #5a483b;
|
|
||||||
--focus-ring: 0 0 0 3px rgba(242, 168, 101, 0.2);
|
|
||||||
}
|
|
||||||
|
|
||||||
body {
|
body {
|
||||||
font-family: 'Inter', -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
|
font-family: 'Inter', -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
|
||||||
background: radial-gradient(120% 120% at 10% 0%, rgba(242, 168, 101, 0.08) 0%, transparent 45%),
|
background: radial-gradient(120% 120% at 12% 0%, rgba(204, 106, 42, 0.08) 0%, transparent 45%),
|
||||||
linear-gradient(180deg, var(--bg-color) 0%, #0f0d0b 100%);
|
linear-gradient(180deg, var(--bg-color) 0%, #f4e9df 100%);
|
||||||
color: var(--text-color);
|
color: var(--text-color);
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
@@ -39,6 +17,12 @@ body {
|
|||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
html.dark body,
|
||||||
|
body.dark {
|
||||||
|
background: radial-gradient(120% 120% at 10% 0%, rgba(242, 168, 101, 0.08) 0%, transparent 45%),
|
||||||
|
linear-gradient(180deg, var(--bg-color) 0%, #0f0d0b 100%);
|
||||||
|
}
|
||||||
|
|
||||||
/* Loading Overlay */
|
/* Loading Overlay */
|
||||||
.loading-overlay {
|
.loading-overlay {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
@@ -74,35 +58,6 @@ body {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
body.light {
|
|
||||||
--bg-color: #fbf6f1;
|
|
||||||
--text-color: #3b2a21;
|
|
||||||
--input-bg: #ffffff;
|
|
||||||
--input-border: #e6d7cc;
|
|
||||||
--button-bg: #f6eee7;
|
|
||||||
--button-hover: #efe3d9;
|
|
||||||
--button-text: #3b2a21;
|
|
||||||
--accent: #cc6a2a;
|
|
||||||
--accent-text: #ffffff;
|
|
||||||
--accent-hover: #e07b36;
|
|
||||||
--highlight-tag: #f2e7dd;
|
|
||||||
--highlight-tag-border: #e7d2c1;
|
|
||||||
--danger: #ef4444;
|
|
||||||
--success: #10b981;
|
|
||||||
--shadow: 0 10px 22px rgba(59, 42, 33, 0.12);
|
|
||||||
--shadow-sm: 0 4px 10px rgba(59, 42, 33, 0.08);
|
|
||||||
--section-bg: #fffaf6;
|
|
||||||
--switch-bg: #e1d5cb;
|
|
||||||
--checkbox-accent: #cc6a2a;
|
|
||||||
--checkbox-border: #d8c8bb;
|
|
||||||
--focus-ring: 0 0 0 3px rgba(204, 106, 42, 0.2);
|
|
||||||
}
|
|
||||||
|
|
||||||
body.light {
|
|
||||||
background: radial-gradient(120% 120% at 12% 0%, rgba(204, 106, 42, 0.08) 0%, transparent 45%),
|
|
||||||
linear-gradient(180deg, var(--bg-color) 0%, #f4e9df 100%);
|
|
||||||
}
|
|
||||||
|
|
||||||
.container {
|
.container {
|
||||||
padding: 14px;
|
padding: 14px;
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -208,27 +163,6 @@ body.light {
|
|||||||
gap: 4px;
|
gap: 4px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.icon-toggle {
|
|
||||||
cursor: pointer;
|
|
||||||
font-size: 16px;
|
|
||||||
color: var(--accent);
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
padding: 4px;
|
|
||||||
border-radius: 8px;
|
|
||||||
border: 1px solid transparent;
|
|
||||||
}
|
|
||||||
|
|
||||||
.icon-toggle:hover {
|
|
||||||
color: var(--accent-hover);
|
|
||||||
background: var(--highlight-tag);
|
|
||||||
border-color: var(--highlight-tag-border);
|
|
||||||
}
|
|
||||||
|
|
||||||
.hidden-toggle {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* GLOBAL HIGHLIGHT ICON: toggle-on/off */
|
/* GLOBAL HIGHLIGHT ICON: toggle-on/off */
|
||||||
.global-icon::before {
|
.global-icon::before {
|
||||||
content: "\f204";
|
content: "\f204";
|
||||||
@@ -238,20 +172,6 @@ body.light {
|
|||||||
content: "\f205";
|
content: "\f205";
|
||||||
}
|
}
|
||||||
|
|
||||||
/* THEME ICON: sun/moon - NO CHECKMARKS */
|
|
||||||
.theme-icon::before {
|
|
||||||
content: "\f185";
|
|
||||||
}
|
|
||||||
|
|
||||||
#themeToggle:checked+.theme-icon::before {
|
|
||||||
content: "\f186";
|
|
||||||
}
|
|
||||||
|
|
||||||
.toggle-icon {
|
|
||||||
font-family: "Font Awesome 6 Free";
|
|
||||||
font-weight: 900;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Sections */
|
/* Sections */
|
||||||
.section {
|
.section {
|
||||||
background: var(--section-bg);
|
background: var(--section-bg);
|
||||||
@@ -300,79 +220,23 @@ body.light {
|
|||||||
opacity: 0.9;
|
opacity: 0.9;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Form Elements */
|
|
||||||
input[type="text"],
|
input[type="text"],
|
||||||
textarea,
|
textarea,
|
||||||
select {
|
select {
|
||||||
width: 100%;
|
|
||||||
padding: 8px 10px;
|
|
||||||
border-radius: 8px;
|
|
||||||
border: 1px solid var(--input-border);
|
|
||||||
background-color: var(--input-bg);
|
|
||||||
color: var(--text-color);
|
|
||||||
font-size: 0.85em;
|
|
||||||
box-sizing: border-box;
|
|
||||||
margin-top: 4px;
|
margin-top: 4px;
|
||||||
margin-bottom: 6px;
|
margin-bottom: 6px;
|
||||||
font-family: inherit;
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="text"]:focus,
|
|
||||||
textarea:focus,
|
|
||||||
select:focus {
|
|
||||||
outline: none;
|
|
||||||
border-color: var(--accent);
|
|
||||||
box-shadow: var(--focus-ring);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
textarea {
|
textarea {
|
||||||
resize: none;
|
resize: none;
|
||||||
height: 60px;
|
height: 60px;
|
||||||
font-size: 0.85em;
|
|
||||||
line-height: 1.4;
|
|
||||||
font-family: inherit;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Color Inputs */
|
|
||||||
input[type="color"] {
|
input[type="color"] {
|
||||||
background: none;
|
|
||||||
border: 1.5px solid var(--input-border);
|
|
||||||
border-radius: 8px;
|
|
||||||
box-shadow: var(--shadow-sm);
|
box-shadow: var(--shadow-sm);
|
||||||
width: 24px;
|
width: 24px;
|
||||||
height: 24px;
|
height: 24px;
|
||||||
margin-left: 6px;
|
margin-left: 6px;
|
||||||
cursor: pointer;
|
|
||||||
padding: 0;
|
|
||||||
appearance: none;
|
|
||||||
-webkit-appearance: none;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="color"]:hover {
|
|
||||||
border-color: var(--accent);
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="color"]::-webkit-color-swatch-wrapper {
|
|
||||||
padding: 0;
|
|
||||||
border-radius: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="color"]::-webkit-color-swatch {
|
|
||||||
border-radius: 0;
|
|
||||||
border: none;
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="color"]::-moz-color-swatch {
|
|
||||||
border-radius: 0;
|
|
||||||
border: none;
|
|
||||||
padding: 0;
|
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-row {
|
.color-row {
|
||||||
@@ -391,132 +255,11 @@ input[type="color"]::-moz-color-swatch {
|
|||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Checkboxes */
|
|
||||||
input[type="checkbox"] {
|
|
||||||
-webkit-appearance: none;
|
|
||||||
-moz-appearance: none;
|
|
||||||
appearance: none;
|
|
||||||
background-color: var(--input-bg);
|
|
||||||
border: 1.5px solid var(--checkbox-border);
|
|
||||||
border-radius: 4px;
|
|
||||||
width: 16px;
|
|
||||||
height: 16px;
|
|
||||||
cursor: pointer;
|
|
||||||
position: relative;
|
|
||||||
flex-shrink: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="checkbox"]:hover {
|
|
||||||
border-color: var(--accent);
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="checkbox"]:checked {
|
|
||||||
background-color: var(--accent);
|
|
||||||
border-color: var(--accent);
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="checkbox"]:checked::before {
|
|
||||||
content: "✓";
|
|
||||||
position: absolute;
|
|
||||||
top: -3px;
|
|
||||||
left: 2px;
|
|
||||||
color: white;
|
|
||||||
font-size: 12px;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Switch Toggle */
|
|
||||||
input[type="checkbox"].switch {
|
|
||||||
-webkit-appearance: none;
|
|
||||||
-moz-appearance: none;
|
|
||||||
appearance: none;
|
|
||||||
width: 36px;
|
|
||||||
height: 20px;
|
|
||||||
background: var(--switch-bg);
|
|
||||||
border: none;
|
|
||||||
border-radius: 20px;
|
|
||||||
position: relative;
|
|
||||||
outline: none;
|
|
||||||
cursor: pointer;
|
|
||||||
transition: background 0.2s;
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="checkbox"].switch::before {
|
|
||||||
content: "";
|
|
||||||
position: absolute;
|
|
||||||
top: 2px;
|
|
||||||
left: 2px;
|
|
||||||
width: 16px;
|
|
||||||
height: 16px;
|
|
||||||
background: white;
|
|
||||||
border-radius: 50%;
|
|
||||||
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.2);
|
|
||||||
transition: left 0.2s;
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="checkbox"].switch:checked {
|
|
||||||
background: var(--accent);
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="checkbox"].switch:checked::before {
|
|
||||||
left: 18px;
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="checkbox"].switch:hover {
|
|
||||||
opacity: 0.9;
|
|
||||||
}
|
|
||||||
|
|
||||||
input[type="checkbox"].switch::after {
|
|
||||||
content: none !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
label:has(input.switch) {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 8px;
|
|
||||||
cursor: pointer;
|
|
||||||
padding: 4px 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Buttons */
|
|
||||||
button {
|
button {
|
||||||
background: var(--button-bg);
|
|
||||||
color: var(--button-text);
|
|
||||||
border: 1px solid var(--input-border);
|
|
||||||
padding: 8px 12px;
|
padding: 8px 12px;
|
||||||
border-radius: 10px;
|
|
||||||
font-weight: 500;
|
|
||||||
cursor: pointer;
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
gap: 5px;
|
|
||||||
font-size: 0.8em;
|
font-size: 0.8em;
|
||||||
font-family: inherit;
|
|
||||||
white-space: nowrap;
|
|
||||||
line-height: 1.2;
|
|
||||||
}
|
|
||||||
|
|
||||||
button i {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
font-size: 0.95em;
|
|
||||||
}
|
|
||||||
|
|
||||||
button:hover {
|
|
||||||
background: var(--button-hover);
|
|
||||||
border-color: var(--highlight-tag-border);
|
|
||||||
}
|
|
||||||
|
|
||||||
button.danger {
|
|
||||||
background: var(--danger);
|
|
||||||
color: white !important;
|
|
||||||
border-color: var(--danger);
|
|
||||||
}
|
|
||||||
|
|
||||||
button.danger:hover {
|
|
||||||
background: #dc2626;
|
|
||||||
border-color: #dc2626;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.button-row {
|
.button-row {
|
||||||
@@ -875,47 +618,7 @@ input[type="file"] {
|
|||||||
border-color: var(--highlight-tag-border);
|
border-color: var(--highlight-tag-border);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Scrollbars */
|
|
||||||
html,
|
|
||||||
body,
|
|
||||||
#wordList,
|
|
||||||
.exceptions-list {
|
|
||||||
scrollbar-width: thin;
|
|
||||||
scrollbar-color: var(--accent) var(--section-bg);
|
|
||||||
}
|
|
||||||
|
|
||||||
html::-webkit-scrollbar,
|
|
||||||
body::-webkit-scrollbar,
|
|
||||||
#wordList::-webkit-scrollbar,
|
|
||||||
.exceptions-list::-webkit-scrollbar {
|
|
||||||
width: 8px;
|
|
||||||
background: var(--section-bg);
|
|
||||||
border-radius: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
html::-webkit-scrollbar-thumb,
|
|
||||||
body::-webkit-scrollbar-thumb,
|
|
||||||
#wordList::-webkit-scrollbar-thumb,
|
|
||||||
.exceptions-list::-webkit-scrollbar-thumb {
|
|
||||||
background: var(--accent);
|
|
||||||
border-radius: 8px;
|
|
||||||
min-height: 24px;
|
|
||||||
border: 2px solid var(--section-bg);
|
|
||||||
}
|
|
||||||
|
|
||||||
html::-webkit-scrollbar-thumb:hover,
|
|
||||||
body::-webkit-scrollbar-thumb:hover,
|
|
||||||
#wordList::-webkit-scrollbar-thumb:hover,
|
|
||||||
.exceptions-list::-webkit-scrollbar-thumb:hover {
|
|
||||||
background: var(--accent-hover);
|
|
||||||
}
|
|
||||||
|
|
||||||
html::-webkit-scrollbar-corner,
|
|
||||||
body::-webkit-scrollbar-corner,
|
|
||||||
#wordList::-webkit-scrollbar-corner,
|
|
||||||
.exceptions-list::-webkit-scrollbar-corner {
|
|
||||||
background: var(--section-bg);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Page Highlights Section */
|
/* Page Highlights Section */
|
||||||
.section[data-section="page-highlights"] {
|
.section[data-section="page-highlights"] {
|
||||||
|
|||||||
@@ -10,7 +10,7 @@
|
|||||||
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@400;600&display=swap" rel="stylesheet">
|
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@400;600&display=swap" rel="stylesheet">
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body class="dark">
|
<body>
|
||||||
<div class="loading-overlay">
|
<div class="loading-overlay">
|
||||||
<div class="loading-spinner"></div>
|
<div class="loading-spinner"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
52
shared/colors.css
Normal file
52
shared/colors.css
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
/* Shared Color Scheme */
|
||||||
|
:root {
|
||||||
|
--bg-color: #fbf6f1;
|
||||||
|
--text-color: #3b2a21;
|
||||||
|
--input-bg: #ffffff;
|
||||||
|
--input-border: #e6d7cc;
|
||||||
|
--button-bg: #f6eee7;
|
||||||
|
--button-hover: #efe3d9;
|
||||||
|
--button-text: #3b2a21;
|
||||||
|
--accent: #cc6a2a;
|
||||||
|
--accent-text: #ffffff;
|
||||||
|
--accent-hover: #e07b36;
|
||||||
|
--highlight-tag: #f2e7dd;
|
||||||
|
--highlight-tag-border: #e7d2c1;
|
||||||
|
--danger: #ef4444;
|
||||||
|
--success: #10b981;
|
||||||
|
--shadow: 0 10px 22px rgba(59, 42, 33, 0.12);
|
||||||
|
--shadow-sm: 0 4px 10px rgba(59, 42, 33, 0.08);
|
||||||
|
--border-radius: 12px;
|
||||||
|
--section-bg: #fffaf6;
|
||||||
|
--panel-bg: #fffaf6;
|
||||||
|
--switch-bg: #e1d5cb;
|
||||||
|
--checkbox-accent: #cc6a2a;
|
||||||
|
--checkbox-border: #d8c8bb;
|
||||||
|
--focus-ring: 0 0 0 3px rgba(204, 106, 42, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark,
|
||||||
|
body.dark {
|
||||||
|
--bg-color: #12100e;
|
||||||
|
--text-color: #f4ede6;
|
||||||
|
--input-bg: #1a1714;
|
||||||
|
--input-border: #3a2e26;
|
||||||
|
--button-bg: #2b211b;
|
||||||
|
--button-hover: #3a2c24;
|
||||||
|
--button-text: #f7efe9;
|
||||||
|
--accent: #f2a865;
|
||||||
|
--accent-hover: #f7c38a;
|
||||||
|
--accent-text: #1b120b;
|
||||||
|
--highlight-tag: #231a14;
|
||||||
|
--highlight-tag-border: #46372c;
|
||||||
|
--danger: #f87171;
|
||||||
|
--success: #4ade80;
|
||||||
|
--shadow: 0 10px 22px rgba(0, 0, 0, 0.5);
|
||||||
|
--shadow-sm: 0 4px 10px rgba(0, 0, 0, 0.4);
|
||||||
|
--section-bg: #17130f;
|
||||||
|
--panel-bg: #17130f;
|
||||||
|
--switch-bg: #4b3a2f;
|
||||||
|
--checkbox-accent: #f2a865;
|
||||||
|
--checkbox-border: #5a483b;
|
||||||
|
--focus-ring: 0 0 0 3px rgba(242, 168, 101, 0.2);
|
||||||
|
}
|
||||||
354
shared/ui-components.css
Normal file
354
shared/ui-components.css
Normal file
@@ -0,0 +1,354 @@
|
|||||||
|
/* Buttons */
|
||||||
|
button {
|
||||||
|
border: none;
|
||||||
|
border-radius: 10px;
|
||||||
|
padding: 8px 10px;
|
||||||
|
background: var(--button-bg);
|
||||||
|
color: var(--button-text);
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
font-family: inherit;
|
||||||
|
font-weight: 500;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 5px;
|
||||||
|
white-space: nowrap;
|
||||||
|
line-height: 1.2;
|
||||||
|
}
|
||||||
|
|
||||||
|
button:hover {
|
||||||
|
background: var(--button-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
button.primary {
|
||||||
|
background: var(--accent);
|
||||||
|
color: var(--accent-text);
|
||||||
|
}
|
||||||
|
|
||||||
|
button.primary:hover {
|
||||||
|
background: var(--accent-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
button.danger {
|
||||||
|
background: rgba(248, 113, 113, 0.15);
|
||||||
|
color: var(--danger);
|
||||||
|
border: 1px solid rgba(248, 113, 113, 0.4);
|
||||||
|
}
|
||||||
|
|
||||||
|
button.danger:hover {
|
||||||
|
background: rgba(248, 113, 113, 0.25);
|
||||||
|
}
|
||||||
|
|
||||||
|
button.ghost {
|
||||||
|
background: transparent;
|
||||||
|
border: 1px solid var(--input-border);
|
||||||
|
}
|
||||||
|
|
||||||
|
button.ghost:hover {
|
||||||
|
background: var(--button-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
button i {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
font-size: 0.95em;
|
||||||
|
}
|
||||||
|
|
||||||
|
button:disabled {
|
||||||
|
opacity: 0.4;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Icon Buttons */
|
||||||
|
.icon-btn {
|
||||||
|
background: transparent;
|
||||||
|
border: none;
|
||||||
|
color: var(--text-color);
|
||||||
|
opacity: 0.6;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 4px 6px;
|
||||||
|
border-radius: 6px;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
min-width: 28px;
|
||||||
|
min-height: 28px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-btn:hover {
|
||||||
|
opacity: 1;
|
||||||
|
background: rgba(242, 168, 101, 0.15);
|
||||||
|
color: var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-btn i {
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Toggle Switches */
|
||||||
|
.toggle-btn {
|
||||||
|
width: 40px;
|
||||||
|
height: 22px;
|
||||||
|
background: rgba(255, 255, 255, 0.1);
|
||||||
|
border: 1px solid var(--input-border);
|
||||||
|
border-radius: 11px;
|
||||||
|
position: relative;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
padding: 0;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toggle-btn::after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
width: 16px;
|
||||||
|
height: 16px;
|
||||||
|
background: var(--text-color);
|
||||||
|
border-radius: 50%;
|
||||||
|
top: 2px;
|
||||||
|
left: 2px;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toggle-btn.active {
|
||||||
|
background: var(--accent);
|
||||||
|
border-color: var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.toggle-btn.active::after {
|
||||||
|
left: 20px;
|
||||||
|
background: var(--accent-text);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Switch Toggle (Checkbox Style) */
|
||||||
|
input[type="checkbox"].switch {
|
||||||
|
-webkit-appearance: none;
|
||||||
|
-moz-appearance: none;
|
||||||
|
appearance: none;
|
||||||
|
width: 36px;
|
||||||
|
height: 20px;
|
||||||
|
background: var(--switch-bg);
|
||||||
|
border: none;
|
||||||
|
border-radius: 20px;
|
||||||
|
position: relative;
|
||||||
|
outline: none;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="checkbox"].switch::before {
|
||||||
|
content: "";
|
||||||
|
position: absolute;
|
||||||
|
top: 2px;
|
||||||
|
left: 2px;
|
||||||
|
width: 16px;
|
||||||
|
height: 16px;
|
||||||
|
background: white;
|
||||||
|
border-radius: 50%;
|
||||||
|
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.2);
|
||||||
|
transition: left 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="checkbox"].switch:checked {
|
||||||
|
background: var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="checkbox"].switch:checked::before {
|
||||||
|
left: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="checkbox"].switch:hover {
|
||||||
|
opacity: 0.9;
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="checkbox"].switch::after {
|
||||||
|
content: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
label:has(input.switch) {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 4px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Checkboxes */
|
||||||
|
input[type="checkbox"]:not(.switch) {
|
||||||
|
-webkit-appearance: none;
|
||||||
|
-moz-appearance: none;
|
||||||
|
appearance: none;
|
||||||
|
background-color: var(--input-bg);
|
||||||
|
border: 1.5px solid var(--checkbox-border);
|
||||||
|
border-radius: 4px;
|
||||||
|
width: 16px;
|
||||||
|
height: 16px;
|
||||||
|
cursor: pointer;
|
||||||
|
position: relative;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="checkbox"]:not(.switch):hover {
|
||||||
|
border-color: var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="checkbox"]:not(.switch):checked {
|
||||||
|
background-color: var(--accent);
|
||||||
|
border-color: var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="checkbox"]:not(.switch):checked::before {
|
||||||
|
content: "✓";
|
||||||
|
position: absolute;
|
||||||
|
top: -3px;
|
||||||
|
left: 2px;
|
||||||
|
color: white;
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Scrollbars */
|
||||||
|
::-webkit-scrollbar {
|
||||||
|
width: 8px;
|
||||||
|
height: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
::-webkit-scrollbar-track {
|
||||||
|
background: var(--section-bg);
|
||||||
|
border-radius: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
::-webkit-scrollbar-thumb {
|
||||||
|
background: var(--accent);
|
||||||
|
border-radius: 8px;
|
||||||
|
min-height: 24px;
|
||||||
|
border: 2px solid var(--section-bg);
|
||||||
|
}
|
||||||
|
|
||||||
|
::-webkit-scrollbar-thumb:hover {
|
||||||
|
background: var(--accent-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
::-webkit-scrollbar-corner {
|
||||||
|
background: var(--section-bg);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Firefox scrollbar */
|
||||||
|
* {
|
||||||
|
scrollbar-width: thin;
|
||||||
|
scrollbar-color: var(--accent) var(--section-bg);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Form Inputs */
|
||||||
|
input[type="text"],
|
||||||
|
textarea,
|
||||||
|
select {
|
||||||
|
width: 100%;
|
||||||
|
padding: 8px 10px;
|
||||||
|
border-radius: 8px;
|
||||||
|
border: 1px solid var(--input-border);
|
||||||
|
background-color: var(--input-bg);
|
||||||
|
color: var(--text-color);
|
||||||
|
font-size: 0.85em;
|
||||||
|
box-sizing: border-box;
|
||||||
|
font-family: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="text"]:focus,
|
||||||
|
textarea:focus,
|
||||||
|
select:focus {
|
||||||
|
outline: none;
|
||||||
|
border-color: var(--accent);
|
||||||
|
box-shadow: var(--focus-ring);
|
||||||
|
}
|
||||||
|
|
||||||
|
textarea {
|
||||||
|
resize: vertical;
|
||||||
|
line-height: 1.4;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Color Inputs */
|
||||||
|
input[type="color"] {
|
||||||
|
background: none;
|
||||||
|
border: 1px solid var(--input-border);
|
||||||
|
border-radius: 8px;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 0;
|
||||||
|
appearance: none;
|
||||||
|
-webkit-appearance: none;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="color"]:hover {
|
||||||
|
border-color: var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="color"]::-webkit-color-swatch-wrapper {
|
||||||
|
padding: 0;
|
||||||
|
border-radius: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="color"]::-webkit-color-swatch {
|
||||||
|
border-radius: 0;
|
||||||
|
border: none;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="color"]::-moz-color-swatch {
|
||||||
|
border-radius: 0;
|
||||||
|
border: none;
|
||||||
|
padding: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Theme Toggle Icon */
|
||||||
|
.icon-toggle {
|
||||||
|
position: relative;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 8px;
|
||||||
|
border-radius: 8px;
|
||||||
|
transition: background-color 0.2s ease;
|
||||||
|
color: var(--accent);
|
||||||
|
font-size: 16px;
|
||||||
|
border: 1px solid transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-toggle:hover {
|
||||||
|
background: var(--highlight-tag);
|
||||||
|
border-color: var(--highlight-tag-border);
|
||||||
|
color: var(--accent-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
.hidden-toggle {
|
||||||
|
position: absolute;
|
||||||
|
opacity: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toggle-icon {
|
||||||
|
font-family: "Font Awesome 6 Free";
|
||||||
|
font-weight: 900;
|
||||||
|
color: inherit;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-icon::before {
|
||||||
|
content: "\f185"; /* sun icon */
|
||||||
|
}
|
||||||
|
|
||||||
|
#themeToggle:checked + .theme-icon::before {
|
||||||
|
content: "\f186"; /* moon icon */
|
||||||
|
}
|
||||||
@@ -10,10 +10,14 @@ export class ListManagerController {
|
|||||||
private selectedWords = new Set<number>();
|
private selectedWords = new Set<number>();
|
||||||
private wordSearchQuery = '';
|
private wordSearchQuery = '';
|
||||||
private isReloading = false;
|
private isReloading = false;
|
||||||
|
private currentPage = 1;
|
||||||
|
private pageSize = 100;
|
||||||
|
private totalWords = 0;
|
||||||
|
|
||||||
async initialize(): Promise<void> {
|
async initialize(): Promise<void> {
|
||||||
await this.loadData();
|
await this.loadData();
|
||||||
this.setupEventListeners();
|
this.setupEventListeners();
|
||||||
|
this.setupTheme();
|
||||||
this.render();
|
this.render();
|
||||||
this.setupStorageSync();
|
this.setupStorageSync();
|
||||||
}
|
}
|
||||||
@@ -56,9 +60,10 @@ export class ListManagerController {
|
|||||||
document.getElementById('moveWordsBtn')?.addEventListener('click', () => this.moveOrCopySelectedWords(false));
|
document.getElementById('moveWordsBtn')?.addEventListener('click', () => this.moveOrCopySelectedWords(false));
|
||||||
document.getElementById('copyWordsBtn')?.addEventListener('click', () => this.moveOrCopySelectedWords(true));
|
document.getElementById('copyWordsBtn')?.addEventListener('click', () => this.moveOrCopySelectedWords(true));
|
||||||
|
|
||||||
const wordSearch = document.getElementById('wordSearch') as HTMLInputElement;
|
const wordSearch = document.getElementById('wordSearch') as HTMLInputElement;
|
||||||
wordSearch.addEventListener('input', (e) => {
|
wordSearch.addEventListener('input', (e) => {
|
||||||
this.wordSearchQuery = (e.target as HTMLInputElement).value;
|
this.wordSearchQuery = (e.target as HTMLInputElement).value;
|
||||||
|
this.currentPage = 1;
|
||||||
this.renderWords();
|
this.renderWords();
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -597,19 +602,25 @@ export class ListManagerController {
|
|||||||
select.disabled = options.length === 0;
|
select.disabled = options.length === 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
private renderWords(): void {
|
private renderWords(): void {
|
||||||
const list = this.lists[this.currentListIndex];
|
const list = this.lists[this.currentListIndex];
|
||||||
const wordList = document.getElementById('wordList');
|
const wordList = document.getElementById('wordList');
|
||||||
if (!list || !wordList) return;
|
if (!list || !wordList) return;
|
||||||
|
|
||||||
const entries = this.getFilteredWordEntries(list);
|
const entries = this.getFilteredWordEntries(list);
|
||||||
|
this.totalWords = entries.length;
|
||||||
|
|
||||||
if (entries.length === 0) {
|
if (entries.length === 0) {
|
||||||
wordList.innerHTML = '<div class="empty">No words in this list.</div>';
|
wordList.innerHTML = '<div class="empty">No words in this list.</div>';
|
||||||
|
this.renderPaginationControls();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
wordList.innerHTML = entries.map(entry => {
|
const startIndex = (this.currentPage - 1) * this.pageSize;
|
||||||
|
const endIndex = Math.min(startIndex + this.pageSize, this.totalWords);
|
||||||
|
const paginatedEntries = entries.slice(startIndex, endIndex);
|
||||||
|
|
||||||
|
wordList.innerHTML = paginatedEntries.map(entry => {
|
||||||
const word = entry.word;
|
const word = entry.word;
|
||||||
const index = entry.index;
|
const index = entry.index;
|
||||||
const isSelected = this.selectedWords.has(index);
|
const isSelected = this.selectedWords.has(index);
|
||||||
@@ -628,13 +639,134 @@ export class ListManagerController {
|
|||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
}).join('');
|
}).join('');
|
||||||
|
|
||||||
|
this.renderPaginationControls();
|
||||||
}
|
}
|
||||||
|
|
||||||
private async save(): Promise<void> {
|
private renderPaginationControls(): void {
|
||||||
|
const paginationContainer = document.getElementById('paginationControls');
|
||||||
|
if (!paginationContainer) return;
|
||||||
|
|
||||||
|
const totalPages = Math.ceil(this.totalWords / this.pageSize);
|
||||||
|
|
||||||
|
if (totalPages <= 1) {
|
||||||
|
paginationContainer.style.display = 'none';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const startItem = (this.currentPage - 1) * this.pageSize + 1;
|
||||||
|
const endItem = Math.min(this.currentPage * this.pageSize, this.totalWords);
|
||||||
|
|
||||||
|
paginationContainer.style.display = 'flex';
|
||||||
|
paginationContainer.innerHTML = `
|
||||||
|
<div class="pagination-info">
|
||||||
|
Showing ${startItem}-${endItem} of ${this.totalWords} words
|
||||||
|
</div>
|
||||||
|
<div class="pagination-controls">
|
||||||
|
<button class="pagination-btn" id="firstPageBtn" ${this.currentPage === 1 ? 'disabled' : ''} title="First page">
|
||||||
|
<i class="fa-solid fa-angles-left"></i>
|
||||||
|
</button>
|
||||||
|
<button class="pagination-btn" id="prevPageBtn" ${this.currentPage === 1 ? 'disabled' : ''} title="Previous page">
|
||||||
|
<i class="fa-solid fa-angle-left"></i>
|
||||||
|
</button>
|
||||||
|
<div class="pagination-pages">
|
||||||
|
<span class="page-info">Page ${this.currentPage} of ${totalPages}</span>
|
||||||
|
</div>
|
||||||
|
<button class="pagination-btn" id="nextPageBtn" ${this.currentPage === totalPages ? 'disabled' : ''} title="Next page">
|
||||||
|
<i class="fa-solid fa-angle-right"></i>
|
||||||
|
</button>
|
||||||
|
<button class="pagination-btn" id="lastPageBtn" ${this.currentPage === totalPages ? 'disabled' : ''} title="Last page">
|
||||||
|
<i class="fa-solid fa-angles-right"></i>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div class="page-size-controls">
|
||||||
|
<label for="pageSizeSelect">Items per page:</label>
|
||||||
|
<select id="pageSizeSelect" class="page-size-select">
|
||||||
|
<option value="25" ${this.pageSize === 25 ? 'selected' : ''}>25</option>
|
||||||
|
<option value="50" ${this.pageSize === 50 ? 'selected' : ''}>50</option>
|
||||||
|
<option value="100" ${this.pageSize === 100 ? 'selected' : ''}>100</option>
|
||||||
|
<option value="200" ${this.pageSize === 200 ? 'selected' : ''}>200</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
|
||||||
|
this.setupPaginationEventListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
private setupPaginationEventListeners(): void {
|
||||||
|
document.getElementById('firstPageBtn')?.addEventListener('click', () => {
|
||||||
|
this.goToPage(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
document.getElementById('prevPageBtn')?.addEventListener('click', () => {
|
||||||
|
this.goToPage(this.currentPage - 1);
|
||||||
|
});
|
||||||
|
|
||||||
|
document.getElementById('nextPageBtn')?.addEventListener('click', () => {
|
||||||
|
this.goToPage(this.currentPage + 1);
|
||||||
|
});
|
||||||
|
|
||||||
|
document.getElementById('lastPageBtn')?.addEventListener('click', () => {
|
||||||
|
const totalPages = Math.ceil(this.totalWords / this.pageSize);
|
||||||
|
this.goToPage(totalPages);
|
||||||
|
});
|
||||||
|
|
||||||
|
const pageSizeSelect = document.getElementById('pageSizeSelect') as HTMLSelectElement;
|
||||||
|
pageSizeSelect?.addEventListener('change', (e) => {
|
||||||
|
const newSize = Number((e.target as HTMLSelectElement).value);
|
||||||
|
if (!Number.isNaN(newSize) && newSize > 0) {
|
||||||
|
this.pageSize = newSize;
|
||||||
|
this.currentPage = 1;
|
||||||
|
this.renderWords();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private goToPage(page: number): void {
|
||||||
|
const totalPages = Math.ceil(this.totalWords / this.pageSize);
|
||||||
|
if (page < 1 || page > totalPages) return;
|
||||||
|
|
||||||
|
this.currentPage = page;
|
||||||
|
this.renderWords();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async save(): Promise<void> {
|
||||||
await StorageService.set({
|
await StorageService.set({
|
||||||
lists: this.lists
|
lists: this.lists
|
||||||
});
|
});
|
||||||
this.render();
|
this.render();
|
||||||
MessageService.sendToAllTabs({ type: 'WORD_LIST_UPDATED' });
|
MessageService.sendToAllTabs({ type: 'WORD_LIST_UPDATED' });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private setupTheme(): void {
|
||||||
|
const toggle = document.getElementById('themeToggle') as HTMLInputElement;
|
||||||
|
const body = document.body;
|
||||||
|
|
||||||
|
const savedTheme = localStorage.getItem('theme');
|
||||||
|
if (savedTheme === 'light') {
|
||||||
|
body.classList.remove('dark');
|
||||||
|
body.classList.add('light');
|
||||||
|
toggle.checked = false;
|
||||||
|
} else {
|
||||||
|
body.classList.add('dark');
|
||||||
|
body.classList.remove('light');
|
||||||
|
toggle.checked = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
toggle.addEventListener('change', () => {
|
||||||
|
if (toggle.checked) {
|
||||||
|
body.classList.add('dark');
|
||||||
|
body.classList.remove('light');
|
||||||
|
document.documentElement.classList.add('dark');
|
||||||
|
document.documentElement.classList.remove('light');
|
||||||
|
localStorage.setItem('theme', 'dark');
|
||||||
|
} else {
|
||||||
|
body.classList.remove('dark');
|
||||||
|
body.classList.add('light');
|
||||||
|
document.documentElement.classList.remove('dark');
|
||||||
|
document.documentElement.classList.add('light');
|
||||||
|
localStorage.setItem('theme', 'light');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,12 @@
|
|||||||
import { ListManagerController } from './ListManagerController.js';
|
import { ListManagerController } from './ListManagerController.js';
|
||||||
|
|
||||||
|
const savedTheme = localStorage.getItem('theme');
|
||||||
|
if (savedTheme === 'light') {
|
||||||
|
document.documentElement.classList.add('light');
|
||||||
|
} else {
|
||||||
|
document.documentElement.classList.add('dark');
|
||||||
|
}
|
||||||
|
|
||||||
document.addEventListener('DOMContentLoaded', async () => {
|
document.addEventListener('DOMContentLoaded', async () => {
|
||||||
const controller = new ListManagerController();
|
const controller = new ListManagerController();
|
||||||
await controller.initialize();
|
await controller.initialize();
|
||||||
|
|||||||
@@ -649,10 +649,14 @@ export class PopupController {
|
|||||||
if (toggle.checked) {
|
if (toggle.checked) {
|
||||||
body.classList.add('dark');
|
body.classList.add('dark');
|
||||||
body.classList.remove('light');
|
body.classList.remove('light');
|
||||||
|
document.documentElement.classList.add('dark');
|
||||||
|
document.documentElement.classList.remove('light');
|
||||||
localStorage.setItem('theme', 'dark');
|
localStorage.setItem('theme', 'dark');
|
||||||
} else {
|
} else {
|
||||||
body.classList.remove('dark');
|
body.classList.remove('dark');
|
||||||
body.classList.add('light');
|
body.classList.add('light');
|
||||||
|
document.documentElement.classList.remove('dark');
|
||||||
|
document.documentElement.classList.add('light');
|
||||||
localStorage.setItem('theme', 'light');
|
localStorage.setItem('theme', 'light');
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,5 +1,12 @@
|
|||||||
import { PopupController } from './PopupController.js';
|
import { PopupController } from './PopupController.js';
|
||||||
|
|
||||||
|
const savedTheme = localStorage.getItem('theme');
|
||||||
|
if (savedTheme === 'light') {
|
||||||
|
document.documentElement.classList.add('light');
|
||||||
|
} else {
|
||||||
|
document.documentElement.classList.add('dark');
|
||||||
|
}
|
||||||
|
|
||||||
function localizePage(): void {
|
function localizePage(): void {
|
||||||
const elements = document.querySelectorAll('[data-i18n]');
|
const elements = document.querySelectorAll('[data-i18n]');
|
||||||
elements.forEach(element => {
|
elements.forEach(element => {
|
||||||
|
|||||||
Reference in New Issue
Block a user