mirror of
				https://github.com/Redocly/redoc.git
				synced 2025-11-04 01:37:32 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			132 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			132 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
import { observer } from 'mobx-react';
 | 
						|
import * as React from 'react';
 | 
						|
 | 
						|
import {
 | 
						|
  ClickablePropertyNameCell,
 | 
						|
  PropertyLabel,
 | 
						|
  RequiredLabel,
 | 
						|
} from '../../common-elements/fields';
 | 
						|
import { FieldDetails } from './FieldDetails';
 | 
						|
import {
 | 
						|
  InnerPropertiesWrap,
 | 
						|
  PropertyBullet,
 | 
						|
  PropertyCellWithInner,
 | 
						|
  PropertyDetailsCell,
 | 
						|
  PropertyNameCell,
 | 
						|
} from '../../common-elements/fields-layout';
 | 
						|
import { ShelfIcon } from '../../common-elements/';
 | 
						|
import { Schema } from '../Schema/Schema';
 | 
						|
 | 
						|
import type { SchemaOptions } from '../Schema/Schema';
 | 
						|
import type { FieldModel } from '../../services/models';
 | 
						|
import { OptionsContext } from '../OptionsProvider';
 | 
						|
import { RedocNormalizedOptions } from '../../services/RedocNormalizedOptions';
 | 
						|
 | 
						|
export interface FieldProps extends SchemaOptions {
 | 
						|
  className?: string;
 | 
						|
  isLast?: boolean;
 | 
						|
  showExamples?: boolean;
 | 
						|
 | 
						|
  field: FieldModel;
 | 
						|
  expandByDefault?: boolean;
 | 
						|
  fieldParentsName?: string[];
 | 
						|
  renderDiscriminatorSwitch?: (opts: FieldProps) => JSX.Element;
 | 
						|
}
 | 
						|
 | 
						|
@observer
 | 
						|
export class Field extends React.Component<FieldProps> {
 | 
						|
  static contextType = OptionsContext;
 | 
						|
  context: RedocNormalizedOptions;
 | 
						|
 | 
						|
  toggle = () => {
 | 
						|
    if (this.props.field.expanded === undefined && this.props.expandByDefault) {
 | 
						|
      this.props.field.collapse();
 | 
						|
    } else {
 | 
						|
      this.props.field.toggle();
 | 
						|
    }
 | 
						|
  };
 | 
						|
 | 
						|
  handleKeyPress = e => {
 | 
						|
    if (e.key === 'Enter') {
 | 
						|
      e.preventDefault();
 | 
						|
      this.toggle();
 | 
						|
    }
 | 
						|
  };
 | 
						|
 | 
						|
  render() {
 | 
						|
    const { hidePropertiesPrefix } = this.context;
 | 
						|
    const { className = '', field, isLast, expandByDefault, fieldParentsName = [] } = this.props;
 | 
						|
    const { name, deprecated, required, kind } = field;
 | 
						|
    const withSubSchema = !field.schema.isPrimitive && !field.schema.isCircular;
 | 
						|
 | 
						|
    const expanded = field.expanded === undefined ? expandByDefault : field.expanded;
 | 
						|
    const labels = (
 | 
						|
      <>
 | 
						|
        {kind === 'additionalProperties' && <PropertyLabel>additional property</PropertyLabel>}
 | 
						|
        {kind === 'patternProperties' && <PropertyLabel>pattern property</PropertyLabel>}
 | 
						|
        {required && <RequiredLabel>required</RequiredLabel>}
 | 
						|
      </>
 | 
						|
    );
 | 
						|
 | 
						|
    const paramName = withSubSchema ? (
 | 
						|
      <ClickablePropertyNameCell
 | 
						|
        className={deprecated ? 'deprecated' : ''}
 | 
						|
        kind={kind}
 | 
						|
        title={name}
 | 
						|
      >
 | 
						|
        <PropertyBullet />
 | 
						|
        <button
 | 
						|
          onClick={this.toggle}
 | 
						|
          onKeyPress={this.handleKeyPress}
 | 
						|
          aria-label={`expand ${name}`}
 | 
						|
        >
 | 
						|
          <ShelfIcon direction={expanded ? 'down' : 'right'} />
 | 
						|
        </button>
 | 
						|
        {!hidePropertiesPrefix &&
 | 
						|
          fieldParentsName.map(
 | 
						|
            name => name + '.\u200B', // zero-width space, a special character is used for correct line breaking
 | 
						|
          )}
 | 
						|
        <span className="property-name" translate="no">{name}</span>
 | 
						|
        {labels}
 | 
						|
      </ClickablePropertyNameCell>
 | 
						|
    ) : (
 | 
						|
      <PropertyNameCell className={deprecated ? 'deprecated' : undefined} kind={kind} title={name}>
 | 
						|
        <PropertyBullet />
 | 
						|
        {!hidePropertiesPrefix &&
 | 
						|
          fieldParentsName.map(
 | 
						|
            name => name + '.\u200B', // zero-width space, a special character is used for correct line breaking
 | 
						|
          )}
 | 
						|
        <span className="property-name" translate="no">{name}</span>
 | 
						|
        {labels}
 | 
						|
      </PropertyNameCell>
 | 
						|
    );
 | 
						|
 | 
						|
    return (
 | 
						|
      <>
 | 
						|
        <tr className={isLast ? 'last ' + className : className}>
 | 
						|
          {paramName}
 | 
						|
          <PropertyDetailsCell>
 | 
						|
            <FieldDetails {...this.props} />
 | 
						|
          </PropertyDetailsCell>
 | 
						|
        </tr>
 | 
						|
        {expanded && withSubSchema && (
 | 
						|
          <tr key={field.name + 'inner'}>
 | 
						|
            <PropertyCellWithInner colSpan={2}>
 | 
						|
              <InnerPropertiesWrap>
 | 
						|
                <Schema
 | 
						|
                  schema={field.schema}
 | 
						|
                  fieldParentsName={[...(fieldParentsName || []), field.name]}
 | 
						|
                  skipReadOnly={this.props.skipReadOnly}
 | 
						|
                  skipWriteOnly={this.props.skipWriteOnly}
 | 
						|
                  showTitle={this.props.showTitle}
 | 
						|
                  level={this.props.level}
 | 
						|
                />
 | 
						|
              </InnerPropertiesWrap>
 | 
						|
            </PropertyCellWithInner>
 | 
						|
          </tr>
 | 
						|
        )}
 | 
						|
      </>
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 |