mirror of
https://github.com/primer/css.git
synced 2025-01-06 22:36:48 +03:00
185 lines
5.5 KiB
JavaScript
185 lines
5.5 KiB
JavaScript
import React from 'react'
|
|
import PropTypes from 'prop-types'
|
|
import chroma from 'chroma-js'
|
|
import colors from 'primer-colors'
|
|
import titleCase from 'title-case'
|
|
import {BorderBox, Box, Flex, Heading, Text} from '@primer/components'
|
|
|
|
const gradientHues = ['gray', 'blue', 'green', 'purple', 'yellow', 'orange', 'red']
|
|
|
|
const {black: BLACK, white: WHITE} = colors
|
|
|
|
export function ColorPalette(props) {
|
|
return (
|
|
<Flex mb={6} className="markdown-no-margin" {...props}>
|
|
{gradientHues.map(hue => {
|
|
const color = colors[hue][5]
|
|
return (
|
|
<Box bg={color} p={3} width={200} mr={2} key={hue}>
|
|
<Text fontWeight="bold" color={overlayColor(color)}>
|
|
{titleCase(hue)}
|
|
</Text>
|
|
</Box>
|
|
)
|
|
})}
|
|
<BorderBox bg="white" p={3} width={200} borderRadius={0}>
|
|
<Text fontWeight="bold" color="black">
|
|
White
|
|
</Text>
|
|
</BorderBox>
|
|
</Flex>
|
|
)
|
|
}
|
|
|
|
export function ColorVariables(props) {
|
|
return (
|
|
<>
|
|
<Flex flexWrap="wrap" className="gutter" {...props}>
|
|
{gradientHues.map(hue => (
|
|
<ColorVariable id={hue} hue={hue} key={hue} />
|
|
))}
|
|
</Flex>
|
|
<Flex flexWrap="wrap" {...props}>
|
|
<FadeVariables id="black" hue="black" bg="black" color="white">
|
|
<BorderBox border={0} borderRadius={0} borderTop={1} borderColor="gray.5" mt={1}>
|
|
<Text is="div" fontSize={2} pt={3} mb={0}>
|
|
Black fades apply alpha transparency to the <Var>$black</Var> variable. The black color value has a slight
|
|
blue hue to match our grays.
|
|
</Text>
|
|
</BorderBox>
|
|
</FadeVariables>
|
|
<FadeVariables id="white" hue="white" over={BLACK}>
|
|
<BorderBox border={0} borderRadius={0} borderTop={1} mt={1}>
|
|
<Text is="div" fontSize={2} pt={3} mb={0}>
|
|
White fades apply alpha transparency to the <Var>$white</Var> variable, below these are shown overlaid on
|
|
a dark gray background.
|
|
</Text>
|
|
</BorderBox>
|
|
</FadeVariables>
|
|
</Flex>
|
|
</>
|
|
)
|
|
}
|
|
|
|
export function ColorVariable({hue, ...rest}) {
|
|
const values = colors[hue]
|
|
return (
|
|
<Flex.Item is={Box} pr={4} mb={6} className="col-6 markdown-no-margin" {...rest}>
|
|
{/* <Heading is="div">{titleCase(hue)}</Heading> */}
|
|
<Box bg={`${hue}.5`} my={2} p={3} color="white">
|
|
<Heading is="div" pb={3} fontSize={56} fontWeight="light">
|
|
{titleCase(hue)}
|
|
</Heading>
|
|
<Flex justifyContent="space-between">
|
|
<Flex.Item flex="1 1 auto" is={Var}>
|
|
${hue}-500
|
|
</Flex.Item>
|
|
<Text fontFamily="mono">{values[5]}</Text>
|
|
</Flex>
|
|
</Box>
|
|
{values.map((value, i) => (
|
|
<Swatch name={`${hue}-${i}00`} value={value} key={value} />
|
|
))}
|
|
</Flex.Item>
|
|
)
|
|
}
|
|
|
|
ColorVariable.propTypes = {
|
|
hue: PropTypes.oneOf(Object.keys(colors)).isRequired
|
|
}
|
|
|
|
export function FadeVariables({hue, color, bg, over, children, ...rest}) {
|
|
const colorValue = colors[hue]
|
|
const alphas = [15, 30, 50, 70, 85]
|
|
const values = alphas.map(alpha => {
|
|
const value = chroma(colorValue)
|
|
.alpha(alpha / 100)
|
|
.css()
|
|
return {
|
|
name: `${hue}-fade-${alpha}`,
|
|
textColor: fadeTextColor(value, over),
|
|
value
|
|
}
|
|
})
|
|
const boxProps = {color, bg}
|
|
return (
|
|
<Flex.Item is={Box} pr={4} mb={6} width={1 / 2} className="markdown-no-margin" {...rest}>
|
|
{/* <Heading is="div">{titleCase(hue)}</Heading> */}
|
|
<Box my={2} p={3} {...boxProps}>
|
|
<Heading is="div" pb={3} fontSize={56} fontWeight="light">
|
|
{titleCase(hue)}
|
|
</Heading>
|
|
<Flex justifyContent="space-between">
|
|
<Flex.Item flex="1 1 auto" is={Var}>
|
|
${hue}
|
|
</Flex.Item>
|
|
<Text fontFamily="mono">
|
|
{chroma(colorValue).css()}
|
|
{' / '}
|
|
{colorValue}
|
|
</Text>
|
|
</Flex>
|
|
{children}
|
|
</Box>
|
|
<Box bg={over}>
|
|
{values.map(swatchProps => (
|
|
<Swatch {...swatchProps} key={swatchProps.name} />
|
|
))}
|
|
</Box>
|
|
</Flex.Item>
|
|
)
|
|
}
|
|
|
|
FadeVariables.propTypes = {
|
|
bg: Box.propTypes.color,
|
|
color: Box.propTypes.color,
|
|
hue: PropTypes.oneOf(['black', 'white']),
|
|
over: PropTypes.string
|
|
}
|
|
|
|
function Swatch(props) {
|
|
const {name, value, textColor = overlayColor(value), ...rest} = props
|
|
return (
|
|
<Box bg={value} color={textColor} {...rest}>
|
|
<Text is={Flex} fontSize={1} justifyContent="space-between">
|
|
<Box p={3}>
|
|
<Var>${name}</Var>
|
|
</Box>
|
|
<Box p={3}>
|
|
<Text fontFamily="mono">{value}</Text>
|
|
</Box>
|
|
</Text>
|
|
</Box>
|
|
)
|
|
}
|
|
|
|
Swatch.propTypes = {
|
|
name: PropTypes.string.isRequired,
|
|
textColor: PropTypes.string,
|
|
value: PropTypes.string.isRequired
|
|
}
|
|
|
|
function Var(props) {
|
|
// FIXME: fontStyle should be a prop, right?
|
|
return <Text is="var" fontWeight="bold" fontFamily="mono" style={{fontStyle: 'normal'}} {...props} />
|
|
}
|
|
|
|
function overlayColor(bg) {
|
|
return chroma(bg).luminance() > 0.5 ? BLACK : WHITE
|
|
}
|
|
|
|
function fadeTextColor(fg, bg = WHITE) {
|
|
const rgb = compositeRGB(fg, bg)
|
|
return overlayColor(rgb)
|
|
}
|
|
|
|
/**
|
|
* Composite ("flatten") a foreground RGBA value with an RGB background into an
|
|
* RGB color for the purposes of measuring contrast or luminance.
|
|
*/
|
|
function compositeRGB(foreground, background) {
|
|
const [fr, fg, fb, fa] = chroma(foreground).rgba()
|
|
const [br, bg, bb] = chroma(background).rgb()
|
|
return chroma([(1 - fa) * br + fa * fr, (1 - fa) * bg + fa * fg, (1 - fa) * bb + fa * fb]).css()
|
|
}
|