mirror of
https://github.com/balena-io/etcher.git
synced 2025-04-24 23:37:18 +00:00
144 lines
3.5 KiB
TypeScript
144 lines
3.5 KiB
TypeScript
/*
|
|
* Copyright 2019 balena.io
|
|
*
|
|
* 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 { Drive as DrivelistDrive } from 'drivelist';
|
|
import * as _ from 'lodash';
|
|
import * as React from 'react';
|
|
import { Txt } from 'rendition';
|
|
import { default as styled } from 'styled-components';
|
|
|
|
import {
|
|
getDriveImageCompatibilityStatuses,
|
|
Image,
|
|
} from '../../../../shared/drive-constraints';
|
|
import { bytesToClosestUnit } from '../../../../shared/units';
|
|
import { getSelectedDrives } from '../../models/selection-state';
|
|
import {
|
|
ChangeButton,
|
|
DetailsText,
|
|
StepButton,
|
|
StepNameButton,
|
|
} from '../../styled-components';
|
|
import { middleEllipsis } from '../../utils/middle-ellipsis';
|
|
|
|
const TargetDetail = styled(props => <Txt.span {...props}></Txt.span>)`
|
|
float: ${({ float }) => float};
|
|
`;
|
|
|
|
interface TargetSelectorProps {
|
|
targets: any[];
|
|
disabled: boolean;
|
|
openDriveSelector: () => any;
|
|
reselectDrive: () => any;
|
|
flashing: boolean;
|
|
show: boolean;
|
|
tooltip: string;
|
|
image: Image;
|
|
}
|
|
|
|
function DriveCompatibilityWarning(props: {
|
|
drive: DrivelistDrive;
|
|
image: Image;
|
|
}) {
|
|
const compatibilityWarnings = getDriveImageCompatibilityStatuses(
|
|
props.drive,
|
|
props.image,
|
|
);
|
|
if (compatibilityWarnings.length === 0) {
|
|
return null;
|
|
}
|
|
const messages = _.map(compatibilityWarnings, 'message');
|
|
return (
|
|
<Txt.span
|
|
className="glyphicon glyphicon-exclamation-sign"
|
|
ml={2}
|
|
tooltip={messages.join(', ')}
|
|
/>
|
|
);
|
|
}
|
|
|
|
export function TargetSelector(props: TargetSelectorProps) {
|
|
const targets = getSelectedDrives();
|
|
|
|
if (targets.length === 1) {
|
|
const target = targets[0];
|
|
return (
|
|
<>
|
|
<StepNameButton plain tooltip={props.tooltip}>
|
|
{middleEllipsis(target.description, 20)}
|
|
</StepNameButton>
|
|
{!props.flashing && (
|
|
<ChangeButton plain mb={14} onClick={props.reselectDrive}>
|
|
Change
|
|
</ChangeButton>
|
|
)}
|
|
<DetailsText>
|
|
<DriveCompatibilityWarning drive={target} image={props.image} />
|
|
{bytesToClosestUnit(target.size)}
|
|
</DetailsText>
|
|
</>
|
|
);
|
|
}
|
|
|
|
if (targets.length > 1) {
|
|
const targetsTemplate = [];
|
|
for (const target of targets) {
|
|
targetsTemplate.push(
|
|
<DetailsText
|
|
key={target.device}
|
|
tooltip={`${target.description} ${
|
|
target.displayName
|
|
} ${bytesToClosestUnit(target.size)}`}
|
|
px={21}
|
|
>
|
|
<Txt.span>
|
|
<DriveCompatibilityWarning drive={target} image={props.image} />
|
|
<TargetDetail float="left">
|
|
{middleEllipsis(target.description, 14)}
|
|
</TargetDetail>
|
|
<TargetDetail float="right">
|
|
{bytesToClosestUnit(target.size)}
|
|
</TargetDetail>
|
|
</Txt.span>
|
|
</DetailsText>,
|
|
);
|
|
}
|
|
return (
|
|
<>
|
|
<StepNameButton plain tooltip={props.tooltip}>
|
|
{targets.length} Targets
|
|
</StepNameButton>
|
|
{!props.flashing && (
|
|
<ChangeButton plain onClick={props.reselectDrive} mb={14}>
|
|
Change
|
|
</ChangeButton>
|
|
)}
|
|
{targetsTemplate}
|
|
</>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<StepButton
|
|
tabindex={targets.length > 0 ? -1 : 2}
|
|
disabled={props.disabled}
|
|
onClick={props.openDriveSelector}
|
|
>
|
|
Select target
|
|
</StepButton>
|
|
);
|
|
}
|