/usr/share/grafana/public/app/features/explore/TraceView/components/TraceTimelineViewer/SpanDetail
// Copyright (c) 2017 Uber Technologies, Inc. // // 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 { css } from '@emotion/css'; import cx from 'classnames'; import * as React from 'react'; import { GrafanaTheme2, TraceKeyValuePair } from '@grafana/data'; import { Counter, Icon, useStyles2 } from '@grafana/ui'; import { autoColor } from '../../Theme'; import TNil from '../../types/TNil'; import * as markers from './AccordianKeyValues.markers'; import KeyValuesTable, { KeyValuesTableLink } from './KeyValuesTable'; import { alignIcon } from '.'; export const getStyles = (theme: GrafanaTheme2) => { return { container: css({ textOverflow: 'ellipsis', }), header: css({ label: 'header', cursor: 'pointer', overflow: 'hidden', padding: '0.25em 0.1em', textOverflow: 'ellipsis', whiteSpace: 'nowrap', '&:hover': { background: autoColor(theme, '#e8e8e8'), }, }), headerLabel: css({ width: '120px', display: 'inline-block', }), headerEmpty: css({ label: 'headerEmpty', background: 'none', cursor: 'initial', }), headerHighContrast: css({ label: 'headerHighContrast', '&:hover': { background: autoColor(theme, '#ddd'), }, }), emptyIcon: css({ label: 'emptyIcon', color: autoColor(theme, '#aaa'), }), summary: css({ label: 'summary', display: 'inline', listStyle: 'none', padding: 0, }), summaryItem: css({ label: 'summaryItem', display: 'inline', paddingRight: '0.5rem', '&:last-child': { paddingRight: 0, borderRight: 'none', }, }), summaryLabel: css({ label: 'summaryLabel', color: autoColor(theme, '#777'), paddingRight: '0.5rem', }), }; }; export type AccordianKeyValuesProps = { className?: string | TNil; data: TraceKeyValuePair[]; logName?: string; highContrast?: boolean; interactive?: boolean; onlyValues?: boolean; showSummary?: boolean; showCountBadge?: boolean; isOpen: boolean; label: string | React.ReactNode; linksGetter?: ((pairs: TraceKeyValuePair[], index: number) => KeyValuesTableLink[]) | TNil; onToggle?: null | (() => void); }; interface KeyValuesSummaryProps { data?: TraceKeyValuePair[] | null; } // export for tests export function KeyValuesSummary({ data = null }: KeyValuesSummaryProps) { const styles = useStyles2(getStyles); if (!Array.isArray(data) || !data.length) { return null; } return ( <ul className={styles.summary}> {data.map((item, i) => ( // `i` is necessary in the key because item.key can repeat <li className={styles.summaryItem} key={`${item.key}-${i}`}> <span className={styles.summaryLabel}>{item.key}</span> {String(item.value)} </li> ))} </ul> ); } export default function AccordianKeyValues({ className = null, data, logName, highContrast = false, interactive = true, isOpen, label, linksGetter, onlyValues = false, showSummary = true, showCountBadge = false, onToggle = null, }: AccordianKeyValuesProps) { const isEmpty = (!Array.isArray(data) || !data.length) && !logName; const styles = useStyles2(getStyles); const iconCls = cx(alignIcon, { [styles.emptyIcon]: isEmpty }); let arrow: React.ReactNode | null = null; let headerProps: {} | null = null; const tableFields = logName ? [{ key: 'event name', value: logName }, ...data] : data; if (interactive) { arrow = isOpen ? ( <Icon name={'angle-down'} className={iconCls} /> ) : ( <Icon name={'angle-right'} className={iconCls} /> ); headerProps = { 'aria-checked': isOpen, onClick: isEmpty ? null : onToggle, role: 'switch', }; } const showDataSummaryFields = showSummary && data.length > 0 && !isOpen; return ( <div className={cx(className, styles.container)}> <div className={cx(styles.header, { [styles.headerEmpty]: isEmpty, [styles.headerHighContrast]: highContrast && !isEmpty, })} {...headerProps} data-testid="AccordianKeyValues--header" > {arrow} <strong data-test={markers.LABEL} className={styles.headerLabel}> {label} {showCountBadge ? <Counter value={data.length} variant="secondary" /> : null} </strong> {showDataSummaryFields && ( <span className={css({ marginLeft: '0.7em' })}> <KeyValuesSummary data={data} /> </span> )} </div> {isOpen && <KeyValuesTable data={tableFields} linksGetter={linksGetter} onlyValues={onlyValues} />} </div> ); }
.
Edit
..
Edit
AccordianKeyValues.markers.tsx
Edit
AccordianKeyValues.test.tsx
Edit
AccordianKeyValues.tsx
Edit
AccordianLogs.test.tsx
Edit
AccordianLogs.tsx
Edit
AccordianReferences.test.tsx
Edit
AccordianReferences.tsx
Edit
DetailState.tsx
Edit
KeyValuesTable.test.tsx
Edit
KeyValuesTable.tsx
Edit
ShareSpanButton.tsx
Edit
SpanDetailLinkButtons.test.tsx
Edit
SpanDetailLinkButtons.tsx
Edit
SpanFlameGraph.tsx
Edit
TextList.test.tsx
Edit
TextList.tsx
Edit
index.test.tsx
Edit
index.tsx
Edit
jsonMarkup.js
Edit