<div id="root"></div>
window.onload = () => {
  const { render } = ReactDOM
const { Component } = React
const { css, injectGlobal, default: styled } = styled
const { min, max, rem, em } = styleUtils

const styleUtils = (function () {
  const css = styled.css

  const rem = px => `${ px / 16 }rem`
  const em = (px, em = 16) => `${ px / em }em`

  const sizes = {
    mobileS: 1,
    mobileM: 375,
    mobileL: 425,

    tabletS: 600,
    tabletM: 768,
    tabletL: 900,

    laptopS: 1024,
    laptopM: 1366,
    laptopL: 1440
  }

  const spacing = {
    xxs: rem(2),
    xs:  rem(4),
    sm:  rem(8),
    md:  rem(16),
    lg:  rem(32),
    xl:  rem(64),
    xxl: rem(128)
  }

  let min = (pxWidth, conditions = '') => (...args) => css`
    @media (min-width: ${ em(pxWidth) }) ${ conditions } {
      ${ css(...args) }
    }
  `

  min = Object.keys(sizes).reduce((min, sizeLabel) => {
    min[sizeLabel] = (...args) => min(sizes[sizeLabel])(...args)
    return min
  }, min)


  let max = (pxWidth, conditions = '') => (...args) => css`
    @media (max-width: ${ em(pxWidth) }) ${ conditions } {
      ${ css(...args) }
    }
  `

  max = Object.keys(sizes).reduce((max, sizeLabel) => {
    max[sizeLabel] = (...args) => max(sizes[sizeLabel])(...args)
    return max
  }, max)

  const addPropSetter = (prop, rules, values = {}) => {
    const getSetRule = val => (rules.set instanceof Function)
      ? rules.set(values[val] || val)
      : rules.set

    if (!prop) return
    if (typeof prop !== 'object') return getSetRule(prop)

    const sizeVal = key => sizes[key] || key
    const sizeLabels = Object.keys(prop)
      .sort((a, b) => sizeVal(a) > sizeVal(b))

    const getRule = val => val === false
      ? rules.unset
      : getSetRule(val)

    return sizeLabels.map(sizeLabel => {
      return min(sizeVal(sizeLabel))([getRule(prop[sizeLabel])])
    })
  }

  const universalProps = css`
    box-sizing: ${ p => p.contentBox ? 'content-box' : 'border-box' };
    ${ p => p.inline && 'display: inline-block;' }
    ${ p => p.relative && 'position: relative;' }
  `

  const marginProps = p => [
    ...addPropSetter(p.m, {
      set: val => `margin: ${ val } !important;`,
      unset: `margin: initial !important;`
    }, spacing),

    ...addPropSetter(p.mx, {
      set: val => `
        margin-right: ${ val } !important;
        margin-left: ${ val } !important;
      `,
      unset: `
        margin-right: initial !important;
        margin-left: initial !important;
      `
    }, spacing),

    ...addPropSetter(p.my, {
      set: val => `
        margin-top: ${ val } !important;
        margin-bottom: ${ val } !important;
      `,
      unset: `
        margin-top: initial !important;
        margin-bottom: initial !important;
      `
    }, spacing),

    ...addPropSetter(p.mt, {
      set: val => `margin-top: ${ val } !important;`,
      unset: `margin-top: initial !important;`
    }, spacing),

    ...addPropSetter(p.mr, {
      set: val => `margin-right: ${ val } !important;`,
      unset: `margin-right: initial !important;`
    }, spacing),

    ...addPropSetter(p.mb, {
      set: val => `margin-bottom: ${ val } !important;`,
      unset: `margin-bottom: initial !important;`
    }, spacing),

    ...addPropSetter(p.ml, {
      set: val => `margin-left: ${ val } !important;`,
      unset: `margin-left: initial !important;`
    }, spacing)
  ]

  const paddingProps = p => [
    ...addPropSetter(p.p, {
      set: val => `padding: ${ val } !important;`,
      unset: `padding: initial !important;`
    }, spacing),

    ...addPropSetter(p.px, {
      set: val => `
        padding-right: ${ val } !important;
        padding-left: ${ val } !important;
      `,
      unset: `
        padding-right: initial !important;
        padding-left: initial !important;
      `
    }, spacing),

    ...addPropSetter(p.py, {
      set: val => `
        padding-top: ${ val } !important;
        padding-bottom: ${ val } !important;
      `,
      unset: `
        padding-top: initial !important;
        padding-bottom: initial !important;
      `
    }, spacing),

    ...addPropSetter(p.pt, {
      set: val => `padding-top: ${ val } !important;`,
      unset: `padding-top: initial !important;`
    }, spacing),

    ...addPropSetter(p.pr, {
      set: val => `padding-right: ${ val } !important;`,
      unset: `padding-right: initial !important;`
    }, spacing),

    ...addPropSetter(p.pb, {
      set: val => `padding-bottom: ${ val } !important;`,
      unset: `padding-bottom: initial !important;`
    }, spacing),

    ...addPropSetter(p.pl, {
      set: val => `padding-left: ${ val } !important;`,
      unset: `padding-left: initial !important;`
    }, spacing)
  ]

  const flexContainerProps = p => [
    ...addPropSetter(p.row, {
      set: `display: flex !important;`,
      unset: `display: initial !important;`
    }),

    ...addPropSetter(p.column, {
      set: `
        display: flex !important;
        flex-direction: column !important;
      `,
      unset: `
        display: initial !important;
        flex-direction: initial !important;
      `
    }),

    ...addPropSetter(p.wrap, {
      set: `flex-wrap: wrap !important;`,
      unset: `flex-wrap: initial !important;`
    }),

    ...addPropSetter(p.wrapReverse, {
      set: `flex-wrap: wrap-reverse !important;`,
      unset: `flex-wrap: initial !important;`
    }),

    ...addPropSetter(p.left, {
      set: `justify-content: flex-start !important;`,
      unset: `justify-content: initial !important;`
    }),

    ...addPropSetter(p.right, {
      set: `justify-content: flex-end !important;`,
      unset: `justify-content: initial !important;`
    }),

    ...addPropSetter(p.center, {
      set: `justify-content: center !important;`,
      unset: `justify-content: initial !important;`
    }),

    ...addPropSetter(p.spaceAround, {
      set: `justify-content: space-around !important;`,
      unset: `justify-content: initial !important;`
    }),

    ...addPropSetter(p.spaceBetween, {
      set: `justify-content: space-between !important;`,
      unset: `justify-content: initial !important;`
    }),

    ...addPropSetter(p.top, {
      set: `align-items: flex-start !important;`,
      unset: `align-items: initial !important;`
    }),

    ...addPropSetter(p.middle, {
      set: `align-items: center !important;`,
      unset: `align-items: initial !important;`
    }),

    ...addPropSetter(p.bottom, {
      set: `align-items: flex-end !important;`,
      unset: `align-items: initial !important;`
    }),

    ...addPropSetter(p.baseline, {
      set: `align-items: baseline !important;`,
      unset: `align-items: initial !important;`
    }),

    ...addPropSetter(p.stretch, {
      set: `align-items: stretch !important;`,
      unset: `align-items: initial !important;`
    }),

    ...addPropSetter(p.contentTop, {
      set: `align-content: flex-start !important;`,
      unset: `align-content: initial !important;`
    }),

    ...addPropSetter(p.contentMiddle, {
      set: `align-content: center !important;`,
      unset: `align-content: initial !important;`
    }),

    ...addPropSetter(p.contentBottom, {
      set: `align-content: flex-end !important;`,
      unset: `align-content: initial !important;`
    }),

    ...addPropSetter(p.contentStretch, {
      set: `align-content: stretch !important;`,
      unset: `align-content: initial !important;`
    }),

    ...addPropSetter(p.contentSpaceAround, {
      set: `align-content: space-around !important;`,
      unset: `align-content: initial !important;`
    }),

    ...addPropSetter(p.contentSpaceBetween, {
      set: `align-content: space-between !important;`,
      unset: `align-content: initial !important;`
    })
  ]

  const flexItemProps = p => [
    ...addPropSetter(p.order, {
      set: val => `order: ${ val } !important;`,
      unset: `order: initial !important;`
    }),

    ...addPropSetter(p.flex, {
      set: val => `flex: ${ val } !important;`,
      unset: `flex: initial !important;`
    }),

    ...addPropSetter(p.flexGrow, {
      set: val => `flex-grow: ${ val } !important;`,
      unset: `flex-grow: initial !important;`
    }),

    ...addPropSetter(p.flexShrink, {
      set: val => `flex-shrink: ${ val } !important;`,
      unset: `flex-shrink: initial !important;`
    }),

    ...addPropSetter(p.flexBasis, {
      set: val => `flex-basis: ${ val } !important;`,
      unset: `flex-basis: initial !important;`
    }),

    ...addPropSetter(p.selfLeft, {
      set: `margin-right: auto !important;`,
      unset: `margin-right: initial !important;`
    }),

    ...addPropSetter(p.selfCenter, {
      set: `
        margin-left: auto !important;
        margin-right: auto !important;
      `,
      unset: `
        margin-left: initial !important;
        margin-right: initial !important;
      `
    }),

    ...addPropSetter(p.selfRight, {
      set: `margin-left: auto !important;`,
      unset: `margin-left: initial !important;`
    }),

    ...addPropSetter(p.selfTop, {
      set: `align-self: flex-start !important;`,
      unset: `align-self: initial !important;`
    }),

    ...addPropSetter(p.selfMiddle, {
      set: `align-self: center !important;`,
      unset: `align-self: initial !important;`
    }),

    ...addPropSetter(p.selfBottom, {
      set: `align-self: flex-end !important;`,
      unset: `align-self: initial !important;`
    }),

    ...addPropSetter(p.selfBaseline, {
      set: `align-self: baseline !important;`,
      unset: `align-self: initial !important;`
    }),

    ...addPropSetter(p.selfStretch, {
      set: `align-self: stretch !important;`,
      unset: `align-self: initial !important;`
    })
  ]

  const computeSpacingRules = (props, getRule) => {
    const spacingLabel = Object.keys(props).filter(label => spacing[label])[0]
    const hasSpacingProp = spacing[spacingLabel] || props.value
    const spacingValue = spacing[spacingLabel] || rem(props.value)
    const spacingRule = getRule(spacingValue)

    const getSizeVal = label => sizes[label] || label
    const getSpacingVal = label => spacing[label] || rem(label)

    const sizesLabelsFromProps =
      Object.keys(props)
        .filter(label => sizes[label])
        .sort((a, b) => getSizeVal(a) > getSizeVal(b))

    const spacingRulesForSizes = sizesLabelsFromProps.map(sizeLabel => {
      const spacingLabel = props[sizeLabel]
      const spacingValue = getSpacingVal(spacingLabel)

      return min(getSizeVal(sizeLabel))`${ getRule(spacingValue) }`
    })

    return css`
      ${ spacingRule }
      ${ spacingRulesForSizes }
    `
  }

  return {
    sizes,
    spacing,
    rem,
    em,
    min,
    max,
    addPropSetter,
    computeSpacingRules,
    universalProps,
    flexContainerProps,
    flexItemProps,
    marginProps,
    paddingProps
  }
})()

injectGlobal`
  body {
    margin: 0;
  }
`

const Wrapper = styled.div`
  box-sizing: border-box;
  display: flex;
  justify-content: space-between;
  align-items: center;
  width: 100%;
  height: 100vh;
  padding: 0 ${rem(50)};
  background-color: dodgerblue;
`

const Link = styled.a`
  -webkit-tap-highlight-color: transparent;
`

const blurShape = "M110,60c0,27.6-22.4,50-50,50c-27.6,0-50-22.4-50-50c0-27.6,22.4-50,50-50C87.6,10,110,32.4,110,60z";
const activeShape = "M96,60c0,27.6-8.4,53-36,53c-27.6,0-36-25.4-36-53c0-27.6,8.4-53,36-53C87.6,7,96,32.4,96,60z";

class Arrow extends Component {
  onActive = () => TweenMax.to(this.circle, .5, {
    attr: { d: activeShape },
    transformOrigin: 'center',
    ease: Elastic.easeOut
  });

  onBlur = () => TweenMax.to(this.circle, 1, {
    attr: { d: blurShape },
    ease: Elastic.easeOut
  });

  render() {
    const { className } = this.props
    const arrowID = Math.random()

    return (
      <div
        className={className}
        onMouseDown={this.onActive}
        onTouchStart={this.onActive}
        onMouseLeave={this.onBlur}
        onMouseUp={this.onBlur}
      >
        <svg
          viewBox="0 0 120 120"
          preserveAspectRatio="none"
          width="100%" height="100%"
        >
          <mask id={`${arrowID}`}>
            <rect fill="#fff" width="100%" height="100%"/>
            <path fill="#000" className="arrow" d="M48.5,61.5v-2.9h17.4l-8-8l2.1-2.1L71.5,60L60,71.5l-2.1-2.1l8-8H48.5z" />
          </mask>
          <path ref={el => this.circle = el} mask={`url(#${arrowID})`} d={blurShape} />
        </svg>
      </div>
    )
  }
}

Arrow = styled(Arrow)`
  width: ${rem(60)};
  height: ${rem(60)};
  fill: #fff;
  cursor: inherit;

  ${min(500)`
    width: ${rem(120)};
    height: ${rem(120)};
  `}

  ${({left}) => left && `transform: rotate(180deg);`}

  .arrow {
    ${max(500)`
      transform-origin: center;
      transform: scale(1.6);
    `}
  }
`

render(
  <Wrapper>
    <Link href="#"><Arrow left /></Link>
    <Link href="#"><Arrow /></Link>
  </Wrapper>,
  document.getElementById('root')
)


}
View Compiled

External CSS

This Pen doesn't use any external CSS resources.

External JavaScript

  1. https://cdnjs.cloudflare.com/ajax/libs/react/15.3.1/react.min.js
  2. https://cdnjs.cloudflare.com/ajax/libs/react/15.3.1/react-dom.min.js
  3. https://unpkg.com/styled-components@2.0.0-2/dist/styled-components.js
  4. https://cdnjs.cloudflare.com/ajax/libs/gsap/1.19.1/TweenMax.min.js