@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/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: var(--radius-sm);
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}