Skip to content

Latest commit

 

History

History
511 lines (365 loc) · 18.7 KB

README.md

File metadata and controls

511 lines (365 loc) · 18.7 KB

Vue Design System (W-I-P)

Vue Design System is an open-source tool for prototyping UI design systems. It provides you and your team a set of organized tools, patterns & practices to build upon, so that you can get started with the actual design system faster.

The tool is built on top of Vue.js, Vue Styleguidist & Vue Webpack Template and is aimed for designers and front-end developers who have at least basic knowledge of component based workflows + HTML, SCSS & JavaScript.

Even though this is a system prototyping tool first and foremost, it has all the means to turn into your organization’s actual design system in the end.

Made by @viljamis and other contributors. See also the official website of Vue Design System.

Screenshot

Features

  • A set of interconnected patterns & practices for you and your team.
  • Global design tokens that can be used inside any component.
  • Simple and sane SVG icon and @font-face defaults.
  • Live reloading for both the app and the system documentation.
  • Automatic generation of documentation (the very page you’re looking at).
  • Documentation and the app logic are separated, so you can have public docs while the app itself is private.
  • Autoprefixing, SCSS, Webpack & CSS extraction included by default (Extend with lint-on-save & unit tests).
  • And more…

Sections

  1. Introduction
  2. Examples
  3. Terminology
  4. Naming of things
  5. Directory structure
  6. Getting started
  7. Working with the system
  8. Editing documentation
  9. Frequently asked questions
  10. Roadmap
  11. Changelog
  12. Authors and license

Examples

Projects using Vue Design System:

Terminology

The following terminology is used throughout Vue Design System. It’s a good idea to get familiar with these concepts before diving deeper.

  • Principles are a set of established design principles that are at the heart of the system. They should guide you when working with Tokens, Elements, Patterns and Templates listed below.
  • Tokens are the visual design atoms of the system. Specifically, they are named entities that store visual design attributes. SalesForce’s system has a great example.
  • Elements are the smallest basic structures of a UI. They can not be broken down any further. Buttons, links and inputs are good examples. Elements utilize Tokens mentioned above.
  • Patterns are UI patterns that fall on the more complex side of the spectrum. Patterns consist of both Elements and Tokens.
  • Templates exist to document the layout and structure of a section or the entirety of an interface. Templates can consist of all three things mentioned above: Patterns, Elements and Tokens.

Hierarchy, visualized:

Template
 ├─ Pattern
 │   ├─ Element
 │   │   ├─ Token
 │   │   └─ Token
 │   └─ Element
 │       └─ Token
 └─ Pattern
     └─ Element
         ├─ Token
         └─ Token

Naming of Things

We use the names to communicate about Tokens, Elements, Patterns and Templates. Hence, they must be short, meaningful and pronounceable. Each name must be:

  • Multiword: Names should be most of the time multi-word, except for root App component: This prevents conflicts with existing and future HTML elements, since all HTML elements are a single word.
  • Meaningful: Not over specific, not overly abstract.
  • Short: Maximum of 2 or 3 words.
  • Pronounceable: We want to be able talk about them.
  • Custom element spec compliant: Don’t use reserved names. Reserved names include:
 * annotation-xml      * color-profile
 * font-face           * font-face-src
 * font-face-uri       * font-face-format
 * font-face-name      * missing-glyph

Prefixing Names

There are a few rules around prefixing things, that you’ll want to follow to keep the system consistent:

  • Tokens: Token names always start with a category name and a hyphen. For example $color- or $space-. If you have subcategories, include them in the naming as well and separate with hyphen, Example: $color-primary- or $color-secondary-.
  • Element, Pattern & Templates: Names don’t have a prefix, but should be multiword or otherwise compatible with existing and future HTML elements.

Directory Structure

Below is the design system’s basic structure. The system’s pieces are split into Tokens, Elements, Patterns and Templates as explained above.

There’s also Styles directory which includes generic functions, mixins and variables that you might need. Currently, there’s for example a set of functions that can create tints and shades out of colors defined in Tokens.

Static directory is meant for any static files like for example icons, fonts and similar.

Router maps Templates to the routes and lets vue-router know when and where to render them.

Finally, docs includes any custom content that is shown in the system documentation.

├─ src
│   ├─ tokens
│   ├─ elements
│   │   └─ icons
│   ├─ patterns
│   ├─ templates
│   │   ├─ Styleguide.vue
│   │   └─ Template 1,2,3…
│   ├─ styles
│   ├─ router
│   └─ App.vue
├─ docs
├─ config
├─ build
└─ static

Getting Started

Vue Design System is built on top of official Vue.js project templates that are to provide opinionated, battery-included development tooling setups so that users can get started with actual app code as fast as possible.

Get started by cloning the repo and installing dependencies:

git clone [email protected]:viljamis/vue-design-system.git
cd vue-design-system
npm install

After installing dependencies, you can start up and serve the project at http://localhost:8080, by running the below command. This will also start up and serve the living documentation at http://localhost:6060.

npm start

Build for production with minification (currently builds only the playground):

npm run build

Build for production and view the bundle analyzer report:

npm run build --report

For a detailed explanation on how things work, check out the Webpack guide and docs for vue-loader.

Working with the System

Tokens

Tokens are the visual design atoms of the design system. Specifically, they are named entities that store visual design attributes. SalesForce’s system has a great example.

Creating a new Token

Creating a new design token is as simple as navigating to /src/tokens/ and editing any of the partials. I strongly recommend you to see what kind of example tokens _spacing.scss or _color.scss already have inside of them.

If you need to add a new category in addition to the existing ones, you’ll have to create a new SCSS partial inside the same directory with a name that starts with _. Once done, you can import the new partial inside /src/tokens/tokens.scss:

/* GLOBAL: DESIGN TOKENS
--------------------------------------------- */

@import "../tokens/color";
@import "../tokens/font";
@import "../tokens/font-size";
@import "../tokens/opacity";
@import "../tokens/line-height";
@import "../tokens/spacing";
@import "../tokens/radius";
@import "../tokens/sizing";
@import "../tokens/shadow";
@import "../tokens/time";
@import "../tokens/media-query";
@import "../tokens/z-index";

Token partials themselves will look somewhat like this, depending on the complexity of your system:

/* SPACE TOKENS
--------------------------------------------- */

$space-tiny: 8px;
$space-small: 16px;
$space-base: 24px;
$space-large: 48px;
$space-x-large: 64px;
$space-xx-large: 128px;
$space-xxx-large: 256px;

Using a Token

Since tokens are imported globally, you can use them inside any Element, Pattern or Template without extra work. Using a token is as simple as:

<style lang="scss" scoped>
  a {
    font-family: $font-primary;
  }
</style>

Elements

Elements are the smallest basic structures of a UI. They can not be broken down any further. Buttons, links, and inputs are good examples. Elements utilize Tokens.

Creating a new Element

To create a new element, you will first want to navigate to /src/elements/ and create a new .vue file. Element names don’t have a prefix, but it’s recommended that they are multiword or otherwise compatible with existing and future HTML elements (to learn more, see Naming of Things).

For the sake of simplicity, let’s imagine you’re creating a button:

First, name your file for example VueButton.vue.

After you’ve created the file, it’s time to get yourself familiar with Vue’s templates and how they work. The basic structure is following:

<template>
  <!-- Your element’s HTML -->
</template>

<script>
  // Your element’s JS
</script>

<style>
  /* Your element’s CSS */
</style>

Looks quite simple, right?

Now, let’s add a little bit of template markup. It’s a button, so we’ll add a basic html <button> and a <slot/> inside of it. Slot is used to allow a parent Pattern to pass DOM elements into a child Element.

<template>
  <button class="button">
    <slot/>
  </button>
</template>

Moving further, we can also add default content for the <slot/> that will be shown if nothing gets passed:

<template>
  <button class="button">
    <slot>I’m a Button!</slot>
  </button>
</template>

<script>
  export default {
    name: 'VueButton',
  };
</script>

<style lang="scss" scoped>
  .button {
    font-family: $font-primary;
    background: $color-primary-rich-black;
    color: $color-primary-white;
  }
</style>

In the above example, I’ve also added some basic style properties which utilize design system’s Tokens. The scoped attribute in <style> means that this SCSS will apply to the current Element only, which is similar to the style encapsulation found in Shadow DOM.

See below for additional examples about passing props to Elements and Patterns.

Example Element with a custom props:
<template>
  <a :href="href">
    <slot/>
  </a>
</template>

<script>
  export default {
    name: 'VueLink',
    props: ['href'],
  };
</script>
Example of a Template utilizing the above Element and props:
<template>
  <vue-link href="https://viljamis.com/">
    This is a label!
  </vue-link>
</template>

<script>
  import VueLink from '@/elements/VueLink';

  export default {
    name: 'MyTemplate',
    components: {
      vueLink: VueLink,
    },
  };
</script>

Using an Element

Now that you’ve created your first custom element, it’s time to use it inside one of the Patterns or Templates. To do this, you will have to first import your element:

<script>
  import VueButton from '@/elements/VueButton';

  export default {
    components: {
      vueButton: VueButton,
    },
  };
</script>

Once imported in <script/>, you can start using the element in your <template/>:

<template>
  <vue-button>
    This content overrides the default slot
  </vue-button>
</template>

As you can see, we’ve also added some basic text content inside our Element to override the default <slot/> contents. If wanted, we could also use the element as is, with the default slot contents shown:

<template>
  <vue-button/>
</template>

Patterns

Patterns are UI patterns that fall on the more complex side of the spectrum. Patterns can consist of both Elements and Tokens.

The exact same rules apply to both Patterns and Elements. From Vue.js’s perspective these are all Vue Components, but for the sake of communication between different teams, diciplines and stakeholders we need a set of unified terms and hierarchy for a system.

To better understand the hierarchy and terms used in this project, see System Hierarchy section.

Templates

Templates exist to document the layout and structure of a section or the entirety of an interface. Templates can consist of Patterns, Elements and Tokens.

Editing Documentation

System’s components are documented based on the comments in your source code and readme files that live inside /docs/. Each Element, Pattern and Template can also include a custom block called <docs/>, which is used to provide an example of the markup. Like this:

<docs>
  ```jsx
  <wrapper>I’m an example that will be shown in docs.</wrapper>
  ```
</docs>

In addition, you can include JSDoc style comment blocks inside your <script/> which will be shown in the documentation. An example:

<script>
  /**
   * A wrapper element is used to wrap elements and patterns.
   */
  export default {
    name: 'Wrapper',
    props: {
      /**
       * The html element name used for the wrapper.
       */
      type: {
        type: String,
        default: 'div'
      }
    }
  }
</script>

Finally, the wrapper Element with all the documentation added:

<template>
  <component :is="type" class="wrapper">
    <slot/>
  </component>
</template>

<script>
  /**
   * A wrapper element is used to wrap elements and patterns.
   */
  export default {
    name: 'Wrapper',
    props: {
      /**
       * The html element name used for the wrapper.
       */
      type: {
        type: String,
        default: 'div'
      }
    }
  }
</script>

<style lang="scss" scoped>
  .wrapper {
    @include reset;
    width: 100%;
    padding: map-get($spacing, large);
  }
</style>

<docs>
  ```jsx
  <wrapper>I’m an example that will be shown in docs.</wrapper>
  ```
</docs>

To learn more about the documentation format and what you can do with it, Please refer to Vue Styleguidist’s official docs on GitHub.

Frequently Asked Questions

I’m working on a new component, but the text is displayed in Comic Sans with red color?

This is the expected behaviour. It’s done to highlight issues with scoping. Elements and Components should have fonts, spacing, etc. defined inside them to keep things as modular as possible. If you need to remove this functionality for some reason, see src/App.vue styles.

I’m trying to use the default Icon element included, but can’t figure out what icons are available?

Currently you can use any icon from Font Awesome. Icons are inside the static/icons directory, so you can also just replace the default SVG icons with your own.

I want to use another typeface, how do I change it?

Vue Design System uses Typekit’s Web Font Loader which is easy to configure. To load your own font files, see Getting Started with WebFontLoader. Currently, the app is loading Fira Sans and a few different weights from Google Fonts. See src/App.vue for an example. To get the fonts working in the Styleguide as well, you’ll want to add the same configuration to docs/styleguide.helper.js.

Why aren’t there more components by default?

Vue Design System is not a front-end component library and never will be. Instead it tries to provide you and your team a set of organized tools, patterns & practices to build upon, so that you can get started with the actual design system faster.

Roadmap

  • Separate documentation from the readme and split onto multiple pages.
  • Add a default Button element with sensible default props for type, style, etc.
  • Better defaults.
  • Better code comments.
  • Use one Webfont config for both the app and the documentation.
  • Use JSON or YAML for Tokens and convert to SCSS (maybe https://github.com/Updater/node-sass-json-importer?)
  • Auto create markup (not just SCSS) for all tokens displayed.
  • Automated accessibility tests.

Changelog

  • 0.1.6 (2017-10-29) - General cleanup, updated readme & fixes to styleguide.
  • 0.1.5 (2017-10-28) - Automatic creation of documentation, better naming conventions, fixes styleguide build scripts and adds better icon workflow.
  • 0.1.2 (2017-10-25) - Fixes styleguide warnings + adds global resets
  • 0.1.1 (2017-10-25) - Adds initial alpha documentation creation
  • 0.1.0 (2017-10-23) - Initial prototype

Authors and License

Viljami Salminen, Artem Sapegin, Rafael Escala, react-styleguidist contributors, vue-styleguidist contributors, Vue.js contributors and vue-webpack-boilerplate contributors.

Licensed under the MIT license.

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.