a0bf6c15f4
Resolves #52 This Pull Request introduces the **NexusSearchBox** search feature with premium unified styling, implements a robust **dynamic Qdrant collection auto-provisioning and batch-vector ingestion pipeline**, integrates a unified **Serilog logging infrastructure** for the Blazor Hybrid environment (MAUI), and resolves the **401 Unauthorized API header propagation error** inside mobile builds. ### 🚀 Key Implementations #### 1. Premium `NexusSearchBox` & Semantic Search UI * **NexusSearchBox Component:** Created an elegant search-as-you-type search box with smooth key navigation, quick-clearing, and seamless dynamic styling. * **Unified Aesthetics:** Refactored the search box isolated styling to align perfectly with the dashboard's design system using glassmorphism, `--nexus-neon` token gradients, and smooth pulse/fade animations. * **Semantic Search Integration:** Integrated semantic search query dispatching (`SearchLibrarySemanticallyQuery`) and wired up navigation seamlessly through the updated `ReaderNavigationService`. * **Tests Hardening:** Added/adapted query assertions in `QueryTests.cs` to guarantee safe parameterization and error boundary mapping. #### 2. Qdrant Collection Provisioning & Vector Ingestion * **Dynamic Auto-Provisioning:** Implemented dynamic checking and lazy-creation of the `knowledge_units` collection using 768 dimensions and Cosine distance. * **High-Performance Ingestion:** Optimized `ProcessKnowledgeUnitsAsync` with high-performance batch embedding generation using `_embeddingGenerator` and deterministic MD5 GUIDs for stable, duplicate-free upsertion. * **Database Cache Clear Sync:** Integrated Qdrant collection deletion in `ClearCacheAsync` to ensure absolute consistency between the PostgreSQL database cache and vector database indices. #### 3. Cross-Platform MAUI Logging (Serilog Infrastructure) * **Serilog Integration:** Configured cross-platform Serilog routing in `SerilogConfiguration.cs`, streaming diagnostic logs safely across native platforms and the Blazor Webview container. * **Interop Bridge:** Built `BlazorLoggingBridge.cs` to capture web console messages and pipe them directly to the native host logger. * **Demo Interface:** Added an interactive `SerilogDemo.razor` sandbox under Pages. #### 4. Resolving 401 Load Errors (Authentication Handler Flow) * **Authentication Header Handler:** Implemented the `MobileAuthenticationHeaderHandler` to correctly extract, validate, and inject bearer JWT tokens into outbound API requests. * **Configuration-based API Host:** Structured standard API URI routing to use clean configuration bindings in `appsettings.json`. --- ### 🧪 Verification & Build Status * Run `dotnet build` from the solution root: Successfully compiled the full multi-targeted solution (`Liczba błędów: 0`). * All unit and integration tests successfully executed and verified (`dotnet test`). --------- Co-authored-by: Marek Jasiński <jasins.marek@gmail.com> Co-authored-by: Marek Jaisński <jasins.marek@gmail.com> Reviewed-on: #51 Co-authored-by: Antigravity <antigravity@google.com> Co-committed-by: Antigravity <antigravity@google.com>
149 lines
3.1 KiB
CSS
149 lines
3.1 KiB
CSS
.nexus-citation-container {
|
|
position: relative;
|
|
display: inline-flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
vertical-align: middle;
|
|
margin: 0 4px;
|
|
}
|
|
|
|
.nexus-citation-trigger {
|
|
background: transparent;
|
|
border: none;
|
|
padding: 0;
|
|
margin: 0;
|
|
cursor: pointer;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
color: #06b6d4; /* Glowing Cyan */
|
|
width: 20px;
|
|
height: 20px;
|
|
position: relative;
|
|
outline: none;
|
|
transition: all 0.3s ease;
|
|
}
|
|
|
|
.nexus-citation-trigger:hover {
|
|
color: #00ff99; /* Neon Green on hover */
|
|
transform: scale(1.2);
|
|
}
|
|
|
|
.neon-radar-svg {
|
|
width: 100%;
|
|
height: 100%;
|
|
filter: drop-shadow(0 0 4px currentColor);
|
|
animation: radar-spin 8s linear infinite;
|
|
}
|
|
|
|
.pulse-ring {
|
|
position: absolute;
|
|
width: 100%;
|
|
height: 100%;
|
|
border: 1px solid currentColor;
|
|
border-radius: 50%;
|
|
animation: radar-ping 1.5s cubic-bezier(0, 0, 0.2, 1) infinite;
|
|
opacity: 0;
|
|
pointer-events: none;
|
|
}
|
|
|
|
.nexus-citation-popup {
|
|
position: absolute;
|
|
bottom: calc(100% + 10px);
|
|
left: 50%;
|
|
transform: translateX(-50%) translateY(5px);
|
|
width: 320px;
|
|
padding: 1rem;
|
|
border-radius: 12px;
|
|
background: rgba(10, 16, 26, 0.9); /* Premium dark background */
|
|
border: 1px solid rgba(6, 182, 212, 0.25); /* Cyan border */
|
|
backdrop-filter: blur(12px);
|
|
-webkit-backdrop-filter: blur(12px);
|
|
box-shadow: 0 10px 25px rgba(0, 0, 0, 0.5), 0 0 12px rgba(6, 182, 212, 0.15);
|
|
z-index: 1000;
|
|
pointer-events: none;
|
|
opacity: 0;
|
|
animation: popup-fade-in 0.2s cubic-bezier(0.16, 1, 0.3, 1) forwards;
|
|
transform-origin: bottom center;
|
|
}
|
|
|
|
.popup-header {
|
|
display: flex;
|
|
align-items: center;
|
|
flex-wrap: wrap;
|
|
gap: 4px;
|
|
font-size: 0.75rem;
|
|
font-weight: 700;
|
|
color: #00ff99; /* Emerald/Neon Green micro-header */
|
|
text-transform: uppercase;
|
|
letter-spacing: 0.5px;
|
|
margin-bottom: 0.5rem;
|
|
border-bottom: 1px solid rgba(255, 255, 255, 0.05);
|
|
padding-bottom: 0.35rem;
|
|
}
|
|
|
|
.separator {
|
|
color: rgba(255, 255, 255, 0.3);
|
|
}
|
|
|
|
.book-title {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 4px;
|
|
}
|
|
|
|
.book-author, .page-number {
|
|
color: rgba(255, 255, 255, 0.6);
|
|
}
|
|
|
|
.popup-body {
|
|
margin-bottom: 0.5rem;
|
|
}
|
|
|
|
.citation-quote {
|
|
font-size: 0.85rem;
|
|
line-height: 1.4;
|
|
color: rgba(255, 255, 255, 0.95);
|
|
font-style: italic;
|
|
margin: 0;
|
|
}
|
|
|
|
.popup-footer {
|
|
display: flex;
|
|
justify-content: flex-end;
|
|
}
|
|
|
|
.id-badge {
|
|
font-size: 0.65rem;
|
|
color: rgba(255, 255, 255, 0.3);
|
|
font-family: monospace;
|
|
}
|
|
|
|
/* Animations */
|
|
@keyframes radar-spin {
|
|
0% { transform: rotate(0deg); }
|
|
100% { transform: rotate(360deg); }
|
|
}
|
|
|
|
@keyframes radar-ping {
|
|
0% {
|
|
transform: scale(1);
|
|
opacity: 0.8;
|
|
}
|
|
100% {
|
|
transform: scale(2.2);
|
|
opacity: 0;
|
|
}
|
|
}
|
|
|
|
@keyframes popup-fade-in {
|
|
0% {
|
|
opacity: 0;
|
|
transform: translateX(-50%) translateY(8px) scale(0.95);
|
|
}
|
|
100% {
|
|
opacity: 1;
|
|
transform: translateX(-50%) translateY(0) scale(1);
|
|
}
|
|
}
|