blob: 65d90707cce51353b5756152af69b8e26c74cd0f [file]
// Copyright (C) 2025 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
@import "./query_builder/builder.scss";
@import "./query_builder/results_panel.scss";
@import "./dashboard/dashboard.scss";
.pf-data-explorer {
height: 100%;
position: relative;
overflow: hidden;
&__header {
display: flex;
align-items: center;
}
&__tabs {
height: 100%;
// Override the default overflow:auto on tab content. The Data Explorer
// manages its own layout (sidebar + canvas) and does not need container
// scrollbars, which can cause scroll jumps when switching tabs.
.pf-tabs__content {
overflow: hidden;
}
}
&__tab-actions {
display: flex;
align-items: center;
}
// Wrapper with nested sub-tabs (Graph | Dashboard). Uses flex column
// so the sub-tab bar gets its natural height and the content fills rest.
&__tab-with-subtabs {
display: flex;
flex-direction: column;
height: 100%;
}
&__sub-tab-bar {
flex-shrink: 0;
padding: 4px 8px;
border-bottom: 1px solid var(--pf-color-border);
}
// Wrapper for each tab's Builder content. The Tabs widget uses Gate
// (display:contents) which has clientHeight === 0; DrawerPanel needs
// a real sized parent to measure its container height.
&__tab-content {
height: 100%;
min-height: 0;
flex: 1;
}
}
// Styles for node documentation (loaded from markdown)
.pf-node-info {
line-height: 1.6;
h1 {
font-size: var(--pf-font-size-xl);
font-weight: 600;
margin-top: 0;
margin-bottom: 1rem;
border-bottom: 1px solid var(--pf-color-border);
padding-bottom: 0.5rem;
}
h2 {
font-size: var(--pf-font-size-l);
font-weight: 600;
margin-top: 1.5rem;
margin-bottom: 0.75rem;
}
h3 {
font-size: var(--pf-font-size-m);
font-weight: 600;
margin-top: 1rem;
margin-bottom: 0.5rem;
}
p {
margin-top: 0;
margin-bottom: 1rem;
}
strong {
font-weight: 600;
}
code {
padding: 0.1rem 0.3rem;
border-radius: 3px;
font-family: var(--monospace-font);
font-size: 0.9em;
}
pre {
padding: 1rem;
border-radius: 4px;
overflow-x: auto;
margin: 1rem 0;
code {
padding: 0;
}
}
ul,
ol {
margin: 0.5rem 0 1rem 1.5rem;
padding-left: 0;
}
li {
margin-bottom: 0.25rem;
}
&-loading {
font-style: italic;
}
&-error {
color: var(--pf-color-danger);
}
}
// Table source specific styling
.pf-table-source-selected {
margin-top: 2rem;
padding-top: 1rem;
border-top: 2px solid var(--pf-color-border);
h2 {
font-size: 1.2em;
font-weight: 600;
margin-top: 0;
margin-bottom: 1rem;
}
.pf-details-box {
padding: 1rem;
border-radius: 4px;
border: 1px solid var(--pf-color-border);
}
}
// Time range source specific styling
.pf-timerange-current-data {
margin-top: 2rem;
padding-top: 1rem;
border-top: 2px solid var(--pf-color-border);
h2 {
font-size: 1.2em;
font-weight: 600;
margin-top: 0;
margin-bottom: 1rem;
}
.pf-table {
width: 100%;
border-collapse: collapse;
margin: 1rem 0;
th,
td {
padding: 0.5rem;
text-align: left;
border: 1px solid var(--pf-color-border);
}
th {
font-weight: 600;
}
}
.pf-timerange-info-mode {
font-style: italic;
margin-top: 0.5rem;
}
}
// Visualisation node details in node graph
.pf-visualisation-node-details {
display: flex;
flex-direction: column;
gap: 4px;
&__config {
display: flex;
align-items: center;
gap: 4px;
}
&__type {
font-weight: 500;
}
&__column {
color: var(--pf-color-text-secondary);
}
}
// Chart view container (for ResultsPanel integration)
.pf-chart-view {
display: flex;
flex-direction: column;
height: 100%;
padding: 16px;
box-sizing: border-box;
&__toolbar {
display: flex;
align-items: center;
justify-content: space-between;
margin-bottom: 12px;
gap: 12px;
flex-shrink: 0;
}
&__title {
font-size: var(--pf-font-size-m);
font-weight: 500;
color: var(--pf-color-text);
}
&__charts {
flex: 1;
min-height: 0;
display: flex;
flex-wrap: wrap;
gap: 16px;
overflow: auto;
// Prevent flex lines from stretching to fill container height.
// Cards have fixed height and should not resize with the panel.
align-content: flex-start;
}
&__column-picker {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
gap: 12px;
width: 100%;
height: 100%;
select {
max-width: 250px;
}
}
// Single chart container (extends Card widget)
&__single {
display: flex;
flex-direction: column;
overflow: visible;
position: relative;
// Fixed height: header (~36px) + content (250px) + padding (8px top)
height: 294px;
flex-shrink: 0;
// Override Card padding
padding: 0;
// Resize handle positioned on right edge
> .pf-resize-handle {
position: absolute;
right: -3px;
top: 0;
bottom: 0;
height: auto;
background: transparent;
// Hide the default line - chart border is sufficient
&::after {
display: none;
}
}
}
&__single-header {
display: flex;
align-items: center;
justify-content: space-between;
padding: 0;
font-size: var(--pf-font-size-s);
font-weight: 500;
color: var(--pf-color-text);
background: var(--pf-color-background-secondary);
border-bottom: 1px solid var(--pf-color-border);
flex-shrink: 0;
}
&__single-actions {
display: flex;
align-items: center;
gap: 4px;
padding-right: 8px;
}
&__single-header-text {
flex: 1;
padding: 8px 12px;
cursor: pointer;
transition: background 0.15s ease;
&:hover {
background: var(--pf-color-hover);
}
}
&__single-header-input {
flex: 1;
padding: 8px 12px;
border: none;
outline: none;
font-size: var(--pf-font-size-s);
font-weight: 500;
font-family: inherit;
background: var(--pf-color-background);
color: var(--pf-color-text);
box-sizing: border-box;
}
// Draggable header styling
&__single-header--draggable {
cursor: grab;
&:active {
cursor: grabbing;
}
}
&__single--dragging {
opacity: 0.5;
}
&__single--drag-over {
border-color: var(--pf-color-primary);
border-style: dashed;
}
&__single-content {
height: 250px; // Fixed height for chart container
flex: none;
display: flex;
align-items: center;
justify-content: center;
padding: 8px 8px 0;
overflow: hidden;
// Datagrid fills the content area edge-to-edge with no padding.
&--datagrid {
padding: 0;
align-items: stretch;
justify-content: stretch;
}
}
}
// Chart grid layouts — charts never grow beyond their basis.
.pf-chart-grid {
&--1,
&--2,
&--2x2,
&--3 {
.pf-chart-view__single {
flex: 0 1 calc(33.333% - 11px);
min-width: 300px;
}
}
}
// Applied when the user resizes a chart via the resize handle.
// Doubled class selector bumps specificity to (0,2,0) so it overrides
// the .pf-chart-grid--X .pf-chart-view__single rules above.
.pf-chart-view__single--custom-width.pf-chart-view__single--custom-width {
width: var(--pf-chart-width);
flex: none;
}
// Chart config popup - horizontal form layout
.pf-chart-config-popup {
// Override form styles for horizontal layout
.pf-form {
display: grid;
grid-template-columns: auto 1fr;
gap: 8px 16px;
align-items: center;
padding: 12px;
min-width: 280px;
}
// FormLabel uses display:contents so its children become grid items
.pf-form__label {
display: contents;
margin-bottom: 0;
// Label text (the span)
> span {
font-size: var(--pf-font-size-s);
color: var(--pf-color-text-secondary);
font-weight: 500;
white-space: nowrap;
}
}
// Inputs take full width of their column
.pf-select {
width: 100%;
min-width: 140px;
margin-bottom: 0;
}
input[type="number"] {
width: 100%;
min-width: 140px;
padding: 6px 8px;
border: 1px solid var(--pf-color-border);
border-radius: 4px;
font-size: var(--pf-font-size-s);
font-family: inherit;
background: var(--pf-color-background);
color: var(--pf-color-text);
box-sizing: border-box;
&:focus {
outline: none;
border-color: var(--pf-color-primary);
}
&::-webkit-inner-spin-button,
&::-webkit-outer-spin-button {
opacity: 1;
}
}
// Button bar spans full width
.pf-form__button-bar {
grid-column: 1 / -1;
margin-top: 4px;
padding-top: 8px;
border-top: 1px solid var(--pf-color-border);
}
}
// Chart cards container in modify panel
.pf-chart-cards-container {
display: flex;
flex-direction: column;
gap: 8px;
}
// Individual chart card (extends Card widget)
.pf-chart-card {
overflow: hidden;
padding: 0;
&--collapsed {
.pf-chart-card__header {
border-bottom: none;
}
}
&__header {
display: flex;
align-items: center;
gap: 8px;
padding: 8px 12px;
background: var(--pf-color-background-secondary);
border-bottom: 1px solid var(--pf-color-border);
cursor: pointer;
user-select: none;
&:hover {
background: var(--pf-color-hover);
}
}
&__toggle {
display: flex;
align-items: center;
justify-content: center;
color: var(--pf-color-text-secondary);
}
&__info {
display: flex;
align-items: center;
gap: 8px;
flex: 1;
min-width: 0;
color: var(--pf-color-text);
.pf-icon {
color: var(--pf-color-primary);
flex-shrink: 0;
}
}
&__title {
font-size: var(--pf-font-size-s);
font-weight: 500;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
cursor: text;
&:hover {
color: var(--pf-color-primary);
}
}
&__title-input {
font-size: var(--pf-font-size-s);
font-weight: 500;
border: none;
background: transparent;
outline: none;
padding: 0;
margin: 0;
font-family: inherit;
color: var(--pf-color-text);
flex: 1;
min-width: 0;
&::placeholder {
color: var(--pf-color-text-muted);
font-style: italic;
}
}
&__actions {
display: flex;
align-items: center;
gap: 4px;
}
&__filter-count {
display: flex;
align-items: center;
gap: 4px;
height: 20px;
padding: 0 4px 0 8px;
border-radius: 10px;
background: var(--pf-color-primary);
color: white;
font-size: 11px;
font-weight: 600;
line-height: 1;
}
&__filter-count-close {
display: flex;
align-items: center;
justify-content: center;
width: 14px;
height: 14px;
border-radius: 50%;
font-size: 12px;
cursor: pointer;
opacity: 0.7;
&:hover {
opacity: 1;
background: rgba(255, 255, 255, 0.3);
}
}
&__body {
display: flex;
flex-direction: column;
gap: 8px;
padding: 12px;
}
&__filter-label {
font-size: var(--pf-font-size-xs);
color: var(--pf-color-text-secondary);
}
&__filters {
display: flex;
flex-wrap: wrap;
gap: 6px;
}
&__empty-message {
color: var(--pf-color-text-muted);
font-size: var(--pf-font-size-s);
font-style: italic;
text-align: center;
padding: 16px 8px;
}
}
// Chart type picker — visual grid of chart types shown when adding a chart.
.pf-chart-type-picker {
display: grid;
grid-template-columns: repeat(3, 1fr);
gap: 8px;
padding: 4px 8px;
&__card {
display: flex;
flex-direction: column;
align-items: center;
gap: 6px;
padding: 12px 6px 8px;
border: 1px solid var(--pf-color-border-secondary);
border-radius: 8px;
background: var(--pf-color-background);
cursor: pointer;
transition:
border-color 0.15s,
box-shadow 0.15s,
background 0.15s;
// Reset button defaults
font: inherit;
color: inherit;
&:hover {
border-color: var(--pf-color-primary);
background: var(--pf-color-surface-highlight);
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
}
&:focus-visible {
outline: 2px solid var(--pf-color-primary);
outline-offset: 2px;
}
&:active {
transform: scale(0.97);
}
&.pf-selected {
border-color: var(--pf-color-primary);
background: var(--pf-color-surface-highlight);
box-shadow: 0 0 0 1px var(--pf-color-primary);
}
}
&__preview {
width: 52px;
height: 40px;
display: flex;
align-items: center;
justify-content: center;
svg {
width: 100%;
height: 100%;
}
}
&__label {
font-size: 11px;
font-weight: 500;
text-align: center;
color: var(--pf-color-text);
white-space: nowrap;
}
&__empty {
padding: 12px 16px;
font-size: 12px;
color: var(--pf-color-text-muted);
text-align: center;
}
}
// Group node — sharp rectangle with strong border
.pf-node.pf-node--group {
background: var(--pf-color-background);
border-radius: 0;
border: 2px solid var(--pf-color-text);
.pf-node-header {
border-radius: 0;
}
&.pf-invalid {
border-color: var(--pf-color-danger);
background: color-mix(
in srgb,
var(--pf-color-danger) 8%,
var(--pf-color-background)
);
}
}
.pf-node-wrapper:has(.pf-node--group) {
border-radius: 0;
}
// Disconnected input node in inner graph preview
.pf-node--disconnected {
opacity: 0.4;
}
// Embedded read-only graph preview for group node side panel
.pf-group-inner-graph {
position: relative;
height: 300px;
border: 1px solid var(--pf-color-border);
border-radius: 4px;
overflow: hidden;
.pf-group-inner-graph__controls {
position: absolute;
bottom: 4px;
right: 4px;
z-index: 1;
}
}
// Read-only column items in group node — reuse draggable-item look without drag
.pf-group-column-readonly {
cursor: default;
.pf-column-type {
cursor: default;
}
}