@package-uploader/ui 1.0.14 → 1.1.1
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.
- package/dist/assets/index-C19M6liw.css +1 -0
- package/dist/assets/index-Ca5beg0c.js +71 -0
- package/dist/index.html +2 -2
- package/package.json +1 -1
- package/src/api/client.ts +129 -3
- package/src/components/CourseStructureStep.tsx +211 -0
- package/src/components/UploadModal.tsx +286 -101
- package/src/index.css +352 -1
- package/src/main.tsx +28 -1
- package/dist/assets/index-8HHKr2PX.css +0 -1
- package/dist/assets/index-Cur4iArP.js +0 -71
package/src/index.css
CHANGED
|
@@ -10,6 +10,7 @@
|
|
|
10
10
|
--color-text: #111827;
|
|
11
11
|
--color-text-muted: #6b7280;
|
|
12
12
|
--radius: 8px;
|
|
13
|
+
--radius-sm: 4px;
|
|
13
14
|
--shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
|
|
14
15
|
--shadow-lg: 0 4px 6px rgba(0, 0, 0, 0.1);
|
|
15
16
|
}
|
|
@@ -1167,7 +1168,7 @@ body {
|
|
|
1167
1168
|
align-items: center;
|
|
1168
1169
|
padding: 0.5rem;
|
|
1169
1170
|
background: var(--color-bg-secondary);
|
|
1170
|
-
border-radius:
|
|
1171
|
+
border-radius: 8px;
|
|
1171
1172
|
}
|
|
1172
1173
|
|
|
1173
1174
|
.upload-file-info {
|
|
@@ -1192,6 +1193,41 @@ body {
|
|
|
1192
1193
|
gap: 0.5rem;
|
|
1193
1194
|
}
|
|
1194
1195
|
|
|
1196
|
+
/* Upload progress bar */
|
|
1197
|
+
.upload-progress-container {
|
|
1198
|
+
display: flex;
|
|
1199
|
+
align-items: center;
|
|
1200
|
+
gap: 0.75rem;
|
|
1201
|
+
min-width: 150px;
|
|
1202
|
+
}
|
|
1203
|
+
|
|
1204
|
+
.upload-progress-bar {
|
|
1205
|
+
flex: 1;
|
|
1206
|
+
height: 8px;
|
|
1207
|
+
background: var(--color-border);
|
|
1208
|
+
border-radius: 4px;
|
|
1209
|
+
overflow: hidden;
|
|
1210
|
+
}
|
|
1211
|
+
|
|
1212
|
+
.upload-progress-fill {
|
|
1213
|
+
height: 100%;
|
|
1214
|
+
background: var(--color-primary);
|
|
1215
|
+
transition: width 0.15s ease-out;
|
|
1216
|
+
}
|
|
1217
|
+
|
|
1218
|
+
.upload-progress-text {
|
|
1219
|
+
font-size: 0.75rem;
|
|
1220
|
+
font-weight: 500;
|
|
1221
|
+
color: var(--color-text-muted);
|
|
1222
|
+
min-width: 36px;
|
|
1223
|
+
text-align: right;
|
|
1224
|
+
}
|
|
1225
|
+
|
|
1226
|
+
.status-processing {
|
|
1227
|
+
color: var(--color-primary);
|
|
1228
|
+
font-size: 0.875rem;
|
|
1229
|
+
}
|
|
1230
|
+
|
|
1195
1231
|
.upload-link-options {
|
|
1196
1232
|
background: var(--color-bg-secondary);
|
|
1197
1233
|
border-radius: var(--radius);
|
|
@@ -1334,3 +1370,318 @@ body {
|
|
|
1334
1370
|
.header .nav {
|
|
1335
1371
|
margin-right: auto;
|
|
1336
1372
|
}
|
|
1373
|
+
|
|
1374
|
+
/* ========================================
|
|
1375
|
+
COURSE STRUCTURE STEP
|
|
1376
|
+
======================================== */
|
|
1377
|
+
|
|
1378
|
+
.course-structure-step {
|
|
1379
|
+
display: flex;
|
|
1380
|
+
flex-direction: column;
|
|
1381
|
+
gap: 0.5rem;
|
|
1382
|
+
}
|
|
1383
|
+
|
|
1384
|
+
.course-structure-header {
|
|
1385
|
+
display: flex;
|
|
1386
|
+
justify-content: space-between;
|
|
1387
|
+
align-items: flex-start;
|
|
1388
|
+
margin-bottom: 0.25rem;
|
|
1389
|
+
}
|
|
1390
|
+
|
|
1391
|
+
.course-structure-header h4 {
|
|
1392
|
+
margin: 0;
|
|
1393
|
+
font-size: 1rem;
|
|
1394
|
+
font-weight: 600;
|
|
1395
|
+
}
|
|
1396
|
+
|
|
1397
|
+
.course-structure-meta {
|
|
1398
|
+
display: flex;
|
|
1399
|
+
align-items: center;
|
|
1400
|
+
gap: 0.5rem;
|
|
1401
|
+
font-size: 0.75rem;
|
|
1402
|
+
color: var(--color-text-muted);
|
|
1403
|
+
}
|
|
1404
|
+
|
|
1405
|
+
.course-structure-badge {
|
|
1406
|
+
background: rgba(37, 99, 235, 0.1);
|
|
1407
|
+
color: var(--color-primary);
|
|
1408
|
+
padding: 0.125rem 0.5rem;
|
|
1409
|
+
border-radius: 999px;
|
|
1410
|
+
font-size: 0.7rem;
|
|
1411
|
+
font-weight: 500;
|
|
1412
|
+
}
|
|
1413
|
+
|
|
1414
|
+
/* Tree nodes */
|
|
1415
|
+
.tree-node {
|
|
1416
|
+
padding: 0.25rem 0;
|
|
1417
|
+
}
|
|
1418
|
+
|
|
1419
|
+
.tree-node-row {
|
|
1420
|
+
display: flex;
|
|
1421
|
+
align-items: center;
|
|
1422
|
+
gap: 0.375rem;
|
|
1423
|
+
min-height: 1.5rem;
|
|
1424
|
+
}
|
|
1425
|
+
|
|
1426
|
+
.tree-node-icon {
|
|
1427
|
+
font-size: 1rem;
|
|
1428
|
+
flex-shrink: 0;
|
|
1429
|
+
}
|
|
1430
|
+
|
|
1431
|
+
.tree-node-label {
|
|
1432
|
+
font-weight: 500;
|
|
1433
|
+
font-size: 0.875rem;
|
|
1434
|
+
flex: 1;
|
|
1435
|
+
}
|
|
1436
|
+
|
|
1437
|
+
.tree-class-input {
|
|
1438
|
+
width: 100%;
|
|
1439
|
+
padding: 0.25rem 0.5rem;
|
|
1440
|
+
font-size: 0.8rem;
|
|
1441
|
+
border: 1px solid var(--color-border);
|
|
1442
|
+
border-radius: var(--radius-sm);
|
|
1443
|
+
background: var(--color-bg);
|
|
1444
|
+
color: var(--color-text);
|
|
1445
|
+
margin-top: 0.25rem;
|
|
1446
|
+
}
|
|
1447
|
+
|
|
1448
|
+
.tree-class-input:focus {
|
|
1449
|
+
outline: none;
|
|
1450
|
+
border-color: var(--color-primary);
|
|
1451
|
+
}
|
|
1452
|
+
|
|
1453
|
+
.tree-class-input::placeholder {
|
|
1454
|
+
color: var(--color-text-muted);
|
|
1455
|
+
opacity: 0.6;
|
|
1456
|
+
}
|
|
1457
|
+
|
|
1458
|
+
.tree-class-input-sm {
|
|
1459
|
+
max-width: 200px;
|
|
1460
|
+
margin-top: 0;
|
|
1461
|
+
margin-left: auto;
|
|
1462
|
+
flex-shrink: 0;
|
|
1463
|
+
}
|
|
1464
|
+
|
|
1465
|
+
/* Course node */
|
|
1466
|
+
.tree-course {
|
|
1467
|
+
background: var(--color-bg-secondary);
|
|
1468
|
+
border: 1px solid var(--color-border);
|
|
1469
|
+
border-radius: var(--radius);
|
|
1470
|
+
padding: 0.5rem 0.75rem;
|
|
1471
|
+
}
|
|
1472
|
+
|
|
1473
|
+
/* Lessons container */
|
|
1474
|
+
.tree-lessons {
|
|
1475
|
+
display: flex;
|
|
1476
|
+
flex-direction: column;
|
|
1477
|
+
gap: 0.125rem;
|
|
1478
|
+
max-height: 400px;
|
|
1479
|
+
overflow-y: auto;
|
|
1480
|
+
border: 1px solid var(--color-border);
|
|
1481
|
+
border-radius: var(--radius);
|
|
1482
|
+
padding: 0.25rem;
|
|
1483
|
+
}
|
|
1484
|
+
|
|
1485
|
+
/* Lesson node */
|
|
1486
|
+
.tree-lesson {
|
|
1487
|
+
border-bottom: 1px solid var(--color-border);
|
|
1488
|
+
}
|
|
1489
|
+
|
|
1490
|
+
.tree-lesson:last-child {
|
|
1491
|
+
border-bottom: none;
|
|
1492
|
+
}
|
|
1493
|
+
|
|
1494
|
+
.tree-node-lesson {
|
|
1495
|
+
cursor: pointer;
|
|
1496
|
+
padding: 0.375rem 0.5rem;
|
|
1497
|
+
border-radius: var(--radius-sm);
|
|
1498
|
+
transition: background 0.15s;
|
|
1499
|
+
}
|
|
1500
|
+
|
|
1501
|
+
.tree-node-lesson:hover {
|
|
1502
|
+
background: var(--color-bg-secondary);
|
|
1503
|
+
}
|
|
1504
|
+
|
|
1505
|
+
.tree-expand {
|
|
1506
|
+
font-size: 0.625rem;
|
|
1507
|
+
width: 1rem;
|
|
1508
|
+
text-align: center;
|
|
1509
|
+
color: var(--color-text-muted);
|
|
1510
|
+
flex-shrink: 0;
|
|
1511
|
+
}
|
|
1512
|
+
|
|
1513
|
+
.tree-lesson-num {
|
|
1514
|
+
color: var(--color-text-muted);
|
|
1515
|
+
font-weight: 600;
|
|
1516
|
+
font-size: 0.75rem;
|
|
1517
|
+
}
|
|
1518
|
+
|
|
1519
|
+
.tree-node-indicator {
|
|
1520
|
+
font-size: 0.65rem;
|
|
1521
|
+
color: var(--color-primary);
|
|
1522
|
+
background: rgba(37, 99, 235, 0.1);
|
|
1523
|
+
padding: 0.0625rem 0.375rem;
|
|
1524
|
+
border-radius: 999px;
|
|
1525
|
+
white-space: nowrap;
|
|
1526
|
+
flex-shrink: 0;
|
|
1527
|
+
}
|
|
1528
|
+
|
|
1529
|
+
/* Lesson expanded content */
|
|
1530
|
+
.tree-lesson-content {
|
|
1531
|
+
padding: 0.25rem 0.5rem 0.5rem 1.5rem;
|
|
1532
|
+
}
|
|
1533
|
+
|
|
1534
|
+
.tree-lesson-class {
|
|
1535
|
+
margin-bottom: 0.375rem;
|
|
1536
|
+
}
|
|
1537
|
+
|
|
1538
|
+
/* Block nodes */
|
|
1539
|
+
.tree-node-block {
|
|
1540
|
+
display: flex;
|
|
1541
|
+
align-items: center;
|
|
1542
|
+
gap: 0.375rem;
|
|
1543
|
+
padding: 0.1875rem 0;
|
|
1544
|
+
}
|
|
1545
|
+
|
|
1546
|
+
.tree-node-block .tree-node-row {
|
|
1547
|
+
flex: 1;
|
|
1548
|
+
min-width: 0;
|
|
1549
|
+
}
|
|
1550
|
+
|
|
1551
|
+
.tree-block-pipe {
|
|
1552
|
+
font-family: monospace;
|
|
1553
|
+
color: var(--color-border);
|
|
1554
|
+
font-size: 0.75rem;
|
|
1555
|
+
flex-shrink: 0;
|
|
1556
|
+
}
|
|
1557
|
+
|
|
1558
|
+
.tree-block-type {
|
|
1559
|
+
font-size: 0.75rem;
|
|
1560
|
+
font-weight: 500;
|
|
1561
|
+
color: var(--color-text);
|
|
1562
|
+
white-space: nowrap;
|
|
1563
|
+
}
|
|
1564
|
+
|
|
1565
|
+
.tree-block-title {
|
|
1566
|
+
font-size: 0.7rem;
|
|
1567
|
+
color: var(--color-text-muted);
|
|
1568
|
+
overflow: hidden;
|
|
1569
|
+
text-overflow: ellipsis;
|
|
1570
|
+
white-space: nowrap;
|
|
1571
|
+
max-width: 200px;
|
|
1572
|
+
}
|
|
1573
|
+
|
|
1574
|
+
/* Parsing indicator */
|
|
1575
|
+
.structure-parsing-indicator {
|
|
1576
|
+
display: flex;
|
|
1577
|
+
align-items: center;
|
|
1578
|
+
gap: 0.75rem;
|
|
1579
|
+
padding: 0.75rem;
|
|
1580
|
+
color: var(--color-primary);
|
|
1581
|
+
font-size: 0.875rem;
|
|
1582
|
+
}
|
|
1583
|
+
|
|
1584
|
+
/* Structure detected badge */
|
|
1585
|
+
.structure-detected-badge {
|
|
1586
|
+
background: rgba(16, 185, 129, 0.1);
|
|
1587
|
+
color: var(--color-success);
|
|
1588
|
+
border: 1px solid var(--color-success);
|
|
1589
|
+
border-radius: var(--radius);
|
|
1590
|
+
padding: 0.5rem 0.75rem;
|
|
1591
|
+
font-size: 0.8rem;
|
|
1592
|
+
}
|
|
1593
|
+
|
|
1594
|
+
/* Multi-file note */
|
|
1595
|
+
.structure-multi-note {
|
|
1596
|
+
font-size: 0.75rem;
|
|
1597
|
+
color: var(--color-text-muted);
|
|
1598
|
+
font-style: italic;
|
|
1599
|
+
padding: 0.25rem 0;
|
|
1600
|
+
}
|
|
1601
|
+
|
|
1602
|
+
/* Class mappings summary on options step */
|
|
1603
|
+
.structure-summary-badge {
|
|
1604
|
+
background: rgba(37, 99, 235, 0.1);
|
|
1605
|
+
color: var(--color-primary);
|
|
1606
|
+
border-radius: var(--radius);
|
|
1607
|
+
padding: 0.5rem 0.75rem;
|
|
1608
|
+
font-size: 0.8rem;
|
|
1609
|
+
margin-top: 0.25rem;
|
|
1610
|
+
}
|
|
1611
|
+
|
|
1612
|
+
/* ========================================
|
|
1613
|
+
COLLAPSIBLE STRUCTURE SECTION
|
|
1614
|
+
======================================== */
|
|
1615
|
+
|
|
1616
|
+
.structure-section {
|
|
1617
|
+
display: flex;
|
|
1618
|
+
flex-direction: column;
|
|
1619
|
+
}
|
|
1620
|
+
|
|
1621
|
+
.structure-toggle-btn {
|
|
1622
|
+
display: flex;
|
|
1623
|
+
align-items: center;
|
|
1624
|
+
gap: 0.5rem;
|
|
1625
|
+
width: 100%;
|
|
1626
|
+
padding: 0.625rem 0.75rem;
|
|
1627
|
+
background: var(--color-bg);
|
|
1628
|
+
border: 1px solid var(--color-border);
|
|
1629
|
+
border-radius: var(--radius);
|
|
1630
|
+
cursor: pointer;
|
|
1631
|
+
font-size: 0.875rem;
|
|
1632
|
+
font-weight: 500;
|
|
1633
|
+
color: var(--color-text);
|
|
1634
|
+
transition: all 0.2s;
|
|
1635
|
+
}
|
|
1636
|
+
|
|
1637
|
+
.structure-toggle-btn:hover:not(:disabled) {
|
|
1638
|
+
border-color: var(--color-primary);
|
|
1639
|
+
background: rgba(37, 99, 235, 0.03);
|
|
1640
|
+
}
|
|
1641
|
+
|
|
1642
|
+
.structure-toggle-btn.open {
|
|
1643
|
+
border-color: var(--color-primary);
|
|
1644
|
+
border-bottom-left-radius: 0;
|
|
1645
|
+
border-bottom-right-radius: 0;
|
|
1646
|
+
background: rgba(37, 99, 235, 0.05);
|
|
1647
|
+
}
|
|
1648
|
+
|
|
1649
|
+
.structure-toggle-btn.disabled {
|
|
1650
|
+
opacity: 0.5;
|
|
1651
|
+
cursor: not-allowed;
|
|
1652
|
+
color: var(--color-text-muted);
|
|
1653
|
+
}
|
|
1654
|
+
|
|
1655
|
+
.structure-toggle-btn .spinner {
|
|
1656
|
+
width: 16px;
|
|
1657
|
+
height: 16px;
|
|
1658
|
+
flex-shrink: 0;
|
|
1659
|
+
}
|
|
1660
|
+
|
|
1661
|
+
.structure-toggle-arrow {
|
|
1662
|
+
margin-left: auto;
|
|
1663
|
+
font-size: 0.625rem;
|
|
1664
|
+
color: var(--color-text-muted);
|
|
1665
|
+
transition: transform 0.2s;
|
|
1666
|
+
}
|
|
1667
|
+
|
|
1668
|
+
.structure-panel {
|
|
1669
|
+
border: 1px solid var(--color-primary);
|
|
1670
|
+
border-top: none;
|
|
1671
|
+
border-bottom-left-radius: var(--radius);
|
|
1672
|
+
border-bottom-right-radius: var(--radius);
|
|
1673
|
+
padding: 0.75rem;
|
|
1674
|
+
background: var(--color-bg);
|
|
1675
|
+
animation: panelSlideDown 0.15s ease-out;
|
|
1676
|
+
}
|
|
1677
|
+
|
|
1678
|
+
@keyframes panelSlideDown {
|
|
1679
|
+
from {
|
|
1680
|
+
opacity: 0;
|
|
1681
|
+
transform: translateY(-4px);
|
|
1682
|
+
}
|
|
1683
|
+
to {
|
|
1684
|
+
opacity: 1;
|
|
1685
|
+
transform: translateY(0);
|
|
1686
|
+
}
|
|
1687
|
+
}
|
package/src/main.tsx
CHANGED
|
@@ -4,9 +4,36 @@ import { BrowserRouter } from 'react-router-dom';
|
|
|
4
4
|
import App from './App';
|
|
5
5
|
import './index.css';
|
|
6
6
|
|
|
7
|
+
// Detect base path at runtime from the current script's location
|
|
8
|
+
// This allows the UI to work at any mount point without rebuild
|
|
9
|
+
function detectBasePath(): string {
|
|
10
|
+
// Try to get from the current script src
|
|
11
|
+
const scripts = document.querySelectorAll('script[src*="assets/index"]');
|
|
12
|
+
if (scripts.length > 0) {
|
|
13
|
+
const src = (scripts[0] as HTMLScriptElement).src;
|
|
14
|
+
const url = new URL(src);
|
|
15
|
+
// Extract path up to /assets/
|
|
16
|
+
const assetsIndex = url.pathname.indexOf('/assets/');
|
|
17
|
+
if (assetsIndex > 0) {
|
|
18
|
+
return url.pathname.substring(0, assetsIndex);
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
// Fallback: use current pathname up to last segment
|
|
22
|
+
const path = window.location.pathname;
|
|
23
|
+
// If path ends with / or has no extension, use it directly
|
|
24
|
+
if (path.endsWith('/')) {
|
|
25
|
+
return path.slice(0, -1) || '/';
|
|
26
|
+
}
|
|
27
|
+
// Otherwise get parent path
|
|
28
|
+
const lastSlash = path.lastIndexOf('/');
|
|
29
|
+
return lastSlash > 0 ? path.substring(0, lastSlash) : '/';
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
const basePath = detectBasePath();
|
|
33
|
+
|
|
7
34
|
ReactDOM.createRoot(document.getElementById('root')!).render(
|
|
8
35
|
<React.StrictMode>
|
|
9
|
-
<BrowserRouter>
|
|
36
|
+
<BrowserRouter basename={basePath}>
|
|
10
37
|
<App />
|
|
11
38
|
</BrowserRouter>
|
|
12
39
|
</React.StrictMode>
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
:root{--color-primary: #2563eb;--color-primary-hover: #1d4ed8;--color-success: #10b981;--color-error: #ef4444;--color-warning: #f59e0b;--color-bg: #ffffff;--color-bg-secondary: #f9fafb;--color-border: #e5e7eb;--color-text: #111827;--color-text-muted: #6b7280;--radius: 8px;--shadow: 0 1px 3px rgba(0, 0, 0, .1);--shadow-lg: 0 4px 6px rgba(0, 0, 0, .1)}*{box-sizing:border-box;margin:0;padding:0}body{font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen,Ubuntu,Cantarell,sans-serif;background:var(--color-bg-secondary);color:var(--color-text);line-height:1.5}.app{min-height:100vh;display:flex;flex-direction:column}.header{background:var(--color-bg);border-bottom:1px solid var(--color-border);padding:1rem 2rem;display:flex;align-items:center;justify-content:space-between}.header h1{font-size:1.25rem;font-weight:600}.nav{display:flex;gap:1rem}.nav a{color:var(--color-text-muted);text-decoration:none;padding:.5rem 1rem;border-radius:var(--radius);transition:all .2s}.nav a:hover,.nav a.active{color:var(--color-primary);background:#2563eb1a}.main{flex:1;padding:2rem;max-width:1200px;margin:0 auto;width:100%}.card{background:var(--color-bg);border:1px solid var(--color-border);border-radius:var(--radius);padding:1.5rem;box-shadow:var(--shadow)}.card+.card{margin-top:1rem}.card-header{display:flex;align-items:center;justify-content:space-between;margin-bottom:1rem}.card-title{font-size:1.125rem;font-weight:600}.btn{display:inline-flex;align-items:center;gap:.5rem;padding:.5rem 1rem;border-radius:var(--radius);border:none;font-size:.875rem;font-weight:500;cursor:pointer;transition:all .2s}.btn-primary{background:var(--color-primary);color:#fff}.btn-primary:hover{background:var(--color-primary-hover)}.btn-secondary{background:var(--color-bg);border:1px solid var(--color-border);color:var(--color-text)}.btn-secondary:hover{background:var(--color-bg-secondary)}.btn:disabled{opacity:.5;cursor:not-allowed}.dropzone{border:2px dashed var(--color-border);border-radius:var(--radius);padding:3rem;text-align:center;cursor:pointer;transition:all .2s}.dropzone:hover,.dropzone.active{border-color:var(--color-primary);background:#2563eb0d}.dropzone p{color:var(--color-text-muted);margin-top:.5rem}.file-list{margin-top:1rem}.file-item{display:flex;align-items:center;justify-content:space-between;padding:.75rem;border:1px solid var(--color-border);border-radius:var(--radius);margin-bottom:.5rem}.file-item .name{font-weight:500}.file-item .size{color:var(--color-text-muted);font-size:.875rem}.file-item .status{display:flex;align-items:center;gap:.5rem}.status-pending{color:var(--color-text-muted)}.status-uploading{color:var(--color-primary)}.status-success{color:var(--color-success)}.status-error{color:var(--color-error)}.document-grid{display:grid;grid-template-columns:repeat(auto-fill,minmax(300px,1fr));gap:1rem}.document-card{background:var(--color-bg);border:1px solid var(--color-border);border-radius:var(--radius);padding:1rem;transition:all .2s}.document-card:hover{box-shadow:var(--shadow-lg)}.document-card .title{font-weight:600;margin-bottom:.25rem}.document-card .meta{color:var(--color-text-muted);font-size:.875rem;margin-bottom:.75rem}.document-card .actions{display:flex;gap:.5rem}.folder-tree{border:1px solid var(--color-border);border-radius:var(--radius);padding:1rem;max-height:400px;overflow-y:auto}.folder-item{padding:.5rem;cursor:pointer;border-radius:var(--radius);display:flex;align-items:center;gap:.5rem}.folder-item:hover{background:var(--color-bg-secondary)}.folder-item.selected{background:#2563eb1a;color:var(--color-primary)}.folder-item .icon{font-size:1.25rem}.modal-overlay{position:fixed;top:0;right:0;bottom:0;left:0;background:#00000080;display:flex;align-items:center;justify-content:center;z-index:1000}.modal{background:var(--color-bg);border-radius:var(--radius);padding:1.5rem;max-width:500px;width:90%;max-height:80vh;overflow-y:auto}.modal-header{display:flex;align-items:center;justify-content:space-between;margin-bottom:1rem}.modal-title{font-size:1.125rem;font-weight:600}.modal-close{background:none;border:none;font-size:1.5rem;cursor:pointer;color:var(--color-text-muted)}.form-group{margin-bottom:1rem}.form-label{display:block;font-weight:500;margin-bottom:.25rem}.form-input,.form-select{width:100%;padding:.5rem;border:1px solid var(--color-border);border-radius:var(--radius);font-size:1rem}.form-input:focus,.form-select:focus{outline:none;border-color:var(--color-primary)}.spinner{width:20px;height:20px;border:2px solid var(--color-border);border-top-color:var(--color-primary);border-radius:50%;animation:spin .8s linear infinite}@keyframes spin{to{transform:rotate(360deg)}}.empty-state{text-align:center;padding:3rem;color:var(--color-text-muted)}.empty-state h3{color:var(--color-text);margin-bottom:.5rem}.alert{padding:1rem;border-radius:var(--radius);margin-bottom:1rem}.alert-success{background:#10b9811a;border:1px solid var(--color-success);color:var(--color-success)}.alert-error{background:#ef44441a;border:1px solid var(--color-error);color:var(--color-error)}.alert-warning{background:#f59e0b1a;border:1px solid var(--color-warning);color:var(--color-warning)}.breadcrumb{display:flex;align-items:center;flex-wrap:wrap;gap:0;font-size:.875rem}.breadcrumb-item{background:none;border:none;color:var(--color-primary);cursor:pointer;padding:.25rem .5rem;border-radius:var(--radius);font-size:inherit}.breadcrumb-item:hover{background:#2563eb1a}.breadcrumb-current{color:var(--color-text);cursor:default}.breadcrumb-current:hover{background:transparent}.breadcrumb-separator{color:var(--color-text-muted);margin:0 .25rem}.browse-page{display:flex;flex-direction:column;gap:1rem}.toolbar{display:flex;justify-content:space-between;align-items:center;background:var(--color-bg);border:1px solid var(--color-border);border-radius:var(--radius);padding:.75rem 1rem;gap:1rem;flex-wrap:wrap}.toolbar-left{display:flex;align-items:center;gap:.5rem;flex:1;min-width:200px}.toolbar-right{display:flex;align-items:center;gap:.5rem}.search-input{padding:.5rem .75rem;border:1px solid var(--color-border);border-radius:var(--radius);font-size:.875rem;width:200px}.search-input:focus{outline:none;border-color:var(--color-primary)}.content-area{background:var(--color-bg);border:1px solid var(--color-border);border-radius:var(--radius);overflow:hidden}.data-grid-container{display:flex;flex-direction:column}.data-grid{width:100%;border-collapse:collapse;font-size:.875rem}.data-grid thead{background:var(--color-bg-secondary);border-bottom:1px solid var(--color-border)}.data-grid th{text-align:left;padding:.75rem 1rem;font-weight:600;color:var(--color-text-muted);white-space:nowrap}.data-grid th.sortable{cursor:pointer;-webkit-user-select:none;user-select:none}.data-grid th.sortable:hover{color:var(--color-text)}.data-grid td{padding:.75rem 1rem;border-bottom:1px solid var(--color-border);vertical-align:middle}.grid-row{transition:background .15s}.grid-row:hover{background:var(--color-bg-secondary)}.folder-row{cursor:pointer}.folder-row:hover{background:#2563eb0d}.col-checkbox{width:40px;text-align:center}.col-name{min-width:250px}.col-format,.col-type{width:120px}.col-updated,.col-created{width:140px}.col-actions{width:100px;text-align:right}.item-icon{margin-right:.5rem;font-size:1rem}.item-name{font-weight:500}.subfolder-count{color:var(--color-text-muted);font-size:.75rem;margin-left:.5rem}.btn-icon{background:none;border:none;padding:.25rem .5rem;cursor:pointer;font-size:1rem;opacity:.6;transition:opacity .15s}.btn-icon:hover{opacity:1}.btn-icon:disabled{opacity:.3;cursor:not-allowed}.btn-icon.pinned,.btn-delete:hover{opacity:1}.data-grid-footer{padding:.75rem 1rem;background:var(--color-bg-secondary);border-top:1px solid var(--color-border);font-size:.75rem;color:var(--color-text-muted)}.data-grid-loading,.data-grid-empty{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:3rem;text-align:center;color:var(--color-text-muted);gap:.5rem}.data-grid-empty .empty-icon{font-size:3rem;margin-bottom:.5rem}.data-grid-empty h3{color:var(--color-text);margin:0}.data-grid-empty p{margin:.5rem 0 1rem}.folder-browser{border:1px solid var(--color-border);border-radius:var(--radius);overflow:hidden}.folder-browser-toolbar{display:flex;justify-content:space-between;align-items:center;padding:.75rem 1rem;background:var(--color-bg-secondary);border-bottom:1px solid var(--color-border)}.folder-browser-actions{display:flex;gap:.5rem}.folder-browser-list{max-height:300px;overflow-y:auto}.folder-browser-table{width:100%;border-collapse:collapse}.folder-browser-row{transition:background .15s}.folder-browser-row:hover{background:var(--color-bg-secondary)}.folder-browser-row.selected{background:#2563eb1a}.folder-browser-row td{padding:.5rem;border-bottom:1px solid var(--color-border);vertical-align:middle}.folder-browser-icon{width:40px;text-align:center;font-size:1.25rem}.folder-browser-name{cursor:pointer;font-weight:500}.folder-browser-name:hover{color:var(--color-primary)}.subfolder-indicator{color:var(--color-text-muted);font-size:.75rem;font-weight:400;margin-left:.5rem}.folder-browser-select{width:100px;text-align:center}.folder-browser-delete{width:50px;text-align:center}.folder-browser-loading,.folder-browser-empty{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:2rem;color:var(--color-text-muted);gap:.75rem}.upload-page{display:flex;flex-direction:column;gap:1rem}.selected-folder-badge{background:#2563eb1a;color:var(--color-primary);padding:.25rem .75rem;border-radius:999px;font-size:.75rem;font-weight:500}.upload-actions{margin-top:1rem;display:flex;gap:.5rem;justify-content:flex-end}.btn-sm{padding:.25rem .5rem;font-size:.75rem}.document-row{cursor:pointer;transition:background .15s}.document-row:hover{background:var(--color-bg-secondary)}.document-row.expanded{background:#2563eb0d}.expand-icon{display:inline-block;width:1rem;margin-right:.25rem;transition:transform .2s;color:var(--color-text-muted)}.expand-icon.expanded{transform:rotate(90deg)}.expanded-row td{padding:0!important;background:var(--color-bg-secondary);border-bottom:2px solid var(--color-border)}.course-detail-panel{padding:1rem;display:flex;flex-direction:column;gap:1rem}.course-detail-loading,.course-detail-error{display:flex;align-items:center;justify-content:center;gap:.75rem;padding:2rem;color:var(--color-text-muted)}.course-detail-error{color:var(--color-error);flex-direction:column}.course-detail-tile{background:var(--color-bg);border:1px solid var(--color-border);border-radius:var(--radius);padding:1rem}.tile-header{display:flex;align-items:center;justify-content:space-between;margin-bottom:1rem;padding-bottom:.75rem;border-bottom:1px solid var(--color-border)}.tile-header h3{font-size:1rem;font-weight:600;margin:0}.btn-close{background:none;border:none;font-size:1.25rem;color:var(--color-text-muted);cursor:pointer;padding:.25rem;line-height:1;border-radius:var(--radius);transition:all .15s}.btn-close:hover{background:var(--color-bg-secondary);color:var(--color-text)}.detail-grid{display:grid;grid-template-columns:repeat(2,1fr);gap:.75rem 1.5rem}.detail-item{display:flex;flex-direction:column;gap:.125rem}.detail-item.span-2{grid-column:span 2}.detail-label{font-size:.7rem;font-weight:600;color:var(--color-text-muted);text-transform:uppercase;letter-spacing:.025em}.detail-value{font-size:.875rem;color:var(--color-text)}.detail-value.mono{font-family:SF Mono,Monaco,Consolas,monospace;font-size:.75rem;background:var(--color-bg-secondary);padding:.125rem .375rem;border-radius:4px}.detail-value.description{line-height:1.4;max-height:4.2em;overflow:hidden;text-overflow:ellipsis}.thin-pack-row{display:grid;grid-template-columns:repeat(2,1fr);gap:1rem}.thin-pack-tile{background:var(--color-bg);border:1px solid var(--color-border);border-radius:var(--radius);padding:1rem;border-left:4px solid}.thin-pack-tile.public{border-left-color:var(--color-success)}.thin-pack-tile.private{border-left-color:var(--color-primary)}.thin-pack-header{display:flex;align-items:flex-start;gap:.75rem;margin-bottom:.75rem}.thin-pack-icon{font-size:1.5rem}.thin-pack-titles{flex:1}.thin-pack-title{font-size:.9rem;font-weight:600;margin:0}.thin-pack-subtitle{font-size:.75rem;color:var(--color-text-muted);margin:.125rem 0 0}.thin-pack-content{display:flex;flex-direction:column;gap:.5rem}.thin-pack-error{font-size:.75rem;color:var(--color-error);padding:.5rem;background:#ef44441a;border-radius:4px}.thin-pack-status{display:flex;align-items:center;gap:.5rem;font-size:.8rem}.thin-pack-status.exists{color:var(--color-success)}.thin-pack-status.exists .status-icon{display:inline-flex;align-items:center;justify-content:center;width:1.25rem;height:1.25rem;background:var(--color-success);color:#fff;border-radius:50%;font-size:.7rem}.thin-pack-status.missing{color:var(--color-text-muted);font-style:italic}.thin-pack-meta{display:flex;gap:1rem;font-size:.7rem;color:var(--color-text-muted)}.thin-pack-actions{display:flex;gap:.5rem;margin-top:.25rem}.thin-pack-slug-input{display:flex;flex-direction:column;gap:.25rem}.thin-pack-slug-input label{font-size:.7rem;color:var(--color-text-muted)}.thin-pack-slug-input input{padding:.375rem .5rem;font-size:.8rem;border:1px solid var(--color-border);border-radius:var(--radius-sm);background:var(--color-bg);color:var(--color-text)}.thin-pack-slug-input input:focus{outline:none;border-color:var(--color-primary)}.thin-pack-slug-input input::placeholder{color:var(--color-text-muted);opacity:.6}.thin-pack-name-input{display:flex;flex-direction:column;gap:.25rem}.thin-pack-name-input label{font-size:.7rem;color:var(--color-text-muted)}.thin-pack-name-input input{padding:.375rem .5rem;font-size:.8rem;border:1px solid var(--color-border);border-radius:var(--radius-sm);background:var(--color-bg);color:var(--color-text)}.thin-pack-name-input input:focus{outline:none;border-color:var(--color-primary)}.thin-pack-name-input input::placeholder{color:var(--color-text-muted);opacity:.6}.token-preview{font-size:.65rem;color:var(--color-text-muted);font-family:monospace;word-break:break-all}.thin-pack-visibility{display:flex;flex-direction:column;gap:.25rem;padding:.5rem;background:var(--color-bg-tertiary);border-radius:4px}.visibility-toggle{display:flex;align-items:center;gap:.5rem;cursor:pointer}.visibility-toggle input[type=checkbox]{width:16px;height:16px;cursor:pointer}.toggle-label{font-size:.85rem;font-weight:500}.visibility-hint{font-size:.7rem;color:var(--color-text-muted);margin-left:24px}.thin-pack-tile.lms{border-left-color:var(--color-success)}.thin-pack-tile.shared{border-left-color:var(--color-primary)}.upload-modal{width:600px;max-width:90vw;max-height:90vh;display:flex;flex-direction:column}.upload-modal-content{flex:1;overflow-y:auto;padding:1rem;display:flex;flex-direction:column;gap:1rem}.upload-modal .dropzone{min-height:120px}.upload-file-list{display:flex;flex-direction:column;gap:.5rem}.upload-file-list h4{margin:0;font-size:.85rem;color:var(--color-text-muted)}.upload-file-item{display:flex;justify-content:space-between;align-items:center;padding:.5rem;background:var(--color-bg-secondary);border-radius:var(--radius-sm)}.upload-file-info{display:flex;flex-direction:column;gap:.125rem}.upload-file-name{font-weight:500;font-size:.875rem}.upload-file-size{font-size:.75rem;color:var(--color-text-muted)}.upload-file-status{display:flex;align-items:center;gap:.5rem}.upload-link-options{background:var(--color-bg-secondary);border-radius:var(--radius);padding:1rem;display:flex;flex-direction:column;gap:.75rem}.upload-link-options h4{margin:0;font-size:.85rem;color:var(--color-text-muted)}.upload-link-option{display:flex;flex-direction:column;gap:.5rem}.upload-link-toggle{display:flex;align-items:center;gap:.5rem;cursor:pointer}.upload-link-toggle input[type=checkbox]{width:1rem;height:1rem;cursor:pointer}.upload-link-toggle span{font-size:.875rem}.upload-link-name-input{display:flex;flex-direction:column;gap:.25rem;padding-left:1.5rem}.upload-link-name-input input{padding:.375rem .5rem;font-size:.8rem;border:1px solid var(--color-border);border-radius:var(--radius-sm);background:var(--color-bg);color:var(--color-text)}.upload-link-name-input input:focus{outline:none;border-color:var(--color-primary)}.upload-link-name-input input::placeholder{color:var(--color-text-muted);opacity:.6}.slug-preview{font-size:.7rem;color:var(--color-text-muted);font-family:monospace}.modal-footer{display:flex;justify-content:flex-end;gap:.5rem;padding:1rem;border-top:1px solid var(--color-border)}@media (max-width: 768px){.detail-grid{grid-template-columns:1fr}.detail-item.span-2{grid-column:span 1}.thin-pack-row{grid-template-columns:1fr}}.user-menu{display:flex;align-items:center;gap:.75rem;margin-left:auto}.user-info{display:flex;align-items:center;gap:.5rem}.user-avatar{width:2rem;height:2rem;border-radius:50%;background:var(--color-primary);color:#fff;display:flex;align-items:center;justify-content:center;font-weight:600;font-size:.875rem}.user-name{font-size:.875rem;color:var(--color-text);max-width:150px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.header{display:flex;align-items:center;gap:1.5rem}.header .nav{margin-right:auto}
|