Skip to content

Instantly share code, notes, and snippets.

Show Gist options
  • Save harunpehlivan/b77317e22a8777d7e14d10771d26a623 to your computer and use it in GitHub Desktop.
Save harunpehlivan/b77317e22a8777d7e14d10771d26a623 to your computer and use it in GitHub Desktop.
angular interactive box-model diagram

angular interactive box-model diagram

angular app to visualize the box-model and see how changing the value of box-sizing affects element size and which properties contrinbute.

A Pen by HARUN PEHLİVAN on CodePen.

License.

<div ng-app="cssBoxModel">
<div ui-view="diagram"></div>
</div>
// code on GitHub
// https://github.com/carolineartz/learning-box-model
// http://car.oline.codes
// rebound of this jQuery demo https://codepen.io/guyroutledge/pen/hgpez
angular.module('cssBoxModel', ['ngAnimate', 'ngSanitize', 'ui.router','ui.slider'])
.config(['$stateProvider', '$locationProvider', function($stateProvider, $locationProvider) {
$stateProvider.state('home', {
url: '',
controller: 'MainCtrl',
views: {
'diagram': {
templateUrl: 'https://s3-us-west-2.amazonaws.com/s.cdpn.io/97151/main.html',
controller: 'MainCtrl'
}
}
});
}])
.directive('labelPositionV', function() {
return {
restrict: 'AE',
scope: {},
link: function(scope, element, attrs, ctrl) {
var id = attrs.id;
attrs.$observe('labelPositionTop', function(value) {
var styleTop = "<style> #" + id + "::before{top:" + ((value / 2.8) - 12) + "px;}</style>";
angular.element(document).find('head').append(styleTop);
});
attrs.$observe('labelPositionBottom', function(value) {
var styleBottom = "<style> #" + id + "::after{bottom:" + ((value / 3) - 13) + "px;}</style>";
angular.element(document).find('head').append(styleBottom);
});
}
};
})
.directive('labelPositionH', function() {
return {
restrict: 'AE',
scope: {},
link: function(scope, element, attrs, ctrl) {
var id = attrs.id;
attrs.$observe('labelPositionRight', function(value) {
var styleRight = "<style> #" + id + "::after{right:" + (1*((value / 2.8) - 12)) + "px;}</style>";
angular.element(document).find('head').append(styleRight);
});
attrs.$observe('labelPositionLeft', function(value) {
var styleLeft = "<style> #" + id + "::before{left:" + (1*((value / 3)-12)) + "px;}</style>";
angular.element(document).find('head').append(styleLeft);
});
}
};
});
// Controller
// **********************************************************************
var app = angular.module('cssBoxModel');
app.controller('MainCtrl', function ($scope) {
//models: default adjustable values
$scope.padding = {
top: 20,
right: 20,
bottom: 20,
left: 20,
v: function () {
return this.top + this.bottom;
},
h: function () {
return this.right + this.left;
}
};
$scope.border = {
top: 15,
right: 15,
bottom: 15,
left: 15,
v: function () {
return this.top + this.bottom;
},
h: function () {
return this.right + this.left;
}
};
$scope.margin = {
top: 20,
right: 20,
bottom: 20,
left: 20,
v: function () {
return this.top + this.bottom;
},
h: function () {
return this.right + this.left;
}
};
$scope.box = {
sizing: 'content-box'
};
$scope.dimensions = {
width: 220,
height: 220
};
$scope.innerContent = {
width: getInnerWidth(),
height: getInnerHeight()
};
$scope.generatedIncludeMargin = false;
$scope.generatedBoxDimensions = {
width: getGeneratedBoxDimensionsWidth(),
height: getGeneratedBoxDimensionsHeight()
};
$scope.checkIncludeMargin = function() {
$scope.generatedBoxDimensions.width = getGeneratedBoxDimensionsWidth();
$scope.generatedBoxDimensions.height = getGeneratedBoxDimensionsHeight();
};
//actual applied style values
$scope.boxPosition = {
left: 50,
top: 56
};
$scope.styleMargin = {
width: 300,
height: 300,
top: -50,
left: -50
};
$scope.styleBorder = {
width: 260,
height: 260,
top: -30,
left: -30
};
$scope.stylePadding = {
width: 242,
height: 242,
top: -20,
left: -20
};
//watch for changes applied to sliders and calculate rendered styles
$scope.$watch(function () {
$scope.boxPosition.top = calcBoxPositionTop() + 6;
$scope.boxPosition.left = calcBoxPositionLeft();
//Margin Styles
$scope.styleMargin.width = $scope.margin.h() + $scope.styleBorder.width;
$scope.styleMargin.height = $scope.margin.v() + $scope.styleBorder.height;
$scope.styleMargin.top = -calcBoxPositionTop();
$scope.styleMargin.left = -calcBoxPositionLeft();
//Border Styles
$scope.styleBorder.width = $scope.border.h() + $scope.stylePadding.width;
$scope.styleBorder.height = $scope.border.v() + $scope.stylePadding.height;
$scope.styleBorder.top = -($scope.border.top + $scope.padding.top);
$scope.styleBorder.left = -($scope.border.left + $scope.padding.left);
//Padding Styles
$scope.stylePadding.width = $scope.padding.h() + getInnerWidth() + 2;
$scope.stylePadding.height = $scope.padding.v() + getInnerHeight() + 2;
$scope.stylePadding.top = -$scope.padding.top;
$scope.stylePadding.left = -$scope.padding.left;
//Inner Content Styles- based on box-sizing
$scope.innerContent.width = getInnerWidth();
$scope.innerContent.height = getInnerHeight();
//Generated Dimensions- based on box-sizing
$scope.generatedBoxDimensions.width = getGeneratedBoxDimensionsWidth();
$scope.generatedBoxDimensions.height = getGeneratedBoxDimensionsHeight();
});
function getInnerWidth() {
var width;
if ($scope.box.sizing === 'border-box') {
width = $scope.dimensions.width -
$scope.border.h() -
$scope.padding.h();
} else {
width = $scope.dimensions.width;
}
return (width > 0) ? width : 0;
}
function getInnerHeight() {
var height;
if ($scope.box.sizing === 'border-box') {
height = $scope.dimensions.height -
$scope.border.v() -
$scope.padding.v();
} else {
height = $scope.dimensions.height;
}
return (height > 0) ? height : 0;
}
//if padding + border > dimension, return (padding + border - dimension) [+ margin]
function getGeneratedBoxDimensionsWidth() {
var width;
if ($scope.box.sizing === 'border-box') {
width = (getInnerWidth() === 0) ? calcPaddingBorderWidth() : $scope.dimensions.width;
} else {
width = $scope.dimensions.width + calcPaddingBorderWidth();
}
return ($scope.generatedIncludeMargin) ? width + $scope.margin.h() : width;
}
function getGeneratedBoxDimensionsHeight() {
var height;
if ($scope.box.sizing === 'border-box') {
height = (getInnerHeight() === 0) ? calcPaddingBorderHeight() : $scope.dimensions.height;
} else {
height = $scope.dimensions.height + calcPaddingBorderHeight();
}
return ($scope.generatedIncludeMargin) ? height + $scope.margin.v() : height;
}
/*
* Private: Helpers
*/
function calcBoxPositionTop() {
return $scope.margin.top + $scope.border.top + $scope.padding.top;
}
function calcBoxPositionLeft() {
return $scope.margin.left + $scope.border.left + $scope.padding.left;
}
function calcPaddingBorderWidth() {
return $scope.padding.h() + $scope.border.h();
}
function calcPaddingBorderHeight() {
return $scope.padding.v() + $scope.border.v();
}
});
@import "neat";
@import "bourbon";
$color-margin: #DE6A63;
$color-padding: #C5D936;
$color-content: #63BCF8;
$color-border: #F8CC63;
$color-box-sizing: #8ADFE0;
$color-bg: #242930;
$color-dark-light: #343434;
$color-dark-lighter: #444;
$color-dark-lightest: #777;
$color-body: #808386;
$color-white: #FFFFFF;
$knob-size: 20px;
$knob-border-width: 3px;
$boxes: (content: $color-content,
padding: $color-padding,
border: $color-border,
margin: $color-margin,
box-sizing: $color-box-sizing,
generated-size: $color-content);
$sliders: content, padding, border, margin;
$lg: new-breakpoint(min-width 1245px 7);
$md: new-breakpoint(min-width 950px);
$sm: new-breakpoint(min-width 790px 6);
$xxs: new-breakpoint(max-width 550px);
$column: golden-ratio(1em, 3) !default; // Column width
$gutter: golden-ratio(1em, 1) !default; // Gutter between each two columns
$grid-columns: 12 !default; // Total number of columns in the grid
$max-width: em(1088) !default; // Max-width of the outer container
$border-box-sizing: true !default; // Makes all elements have a border-box layout
$default-feature: min-width; // Default @media feature for the breakpoint() mixin
$default-layout-direction: LTR !default;
@mixin media($query:$feature $value $columns, $total-columns: $grid-columns) {
@if length($query) == 1 {
@media screen and ($default-feature: nth($query, 1)) {
$default-grid-columns: $grid-columns;
$grid-columns: $total-columns;
@content;
$grid-columns: $default-grid-columns;
}
}
@else if length($query) == 2 {
@media screen and (nth($query, 1): nth($query, 2)) {
$default-grid-columns: $grid-columns;
$grid-columns: $total-columns;
@content;
$grid-columns: $default-grid-columns;
}
}
@else if length($query) == 3 {
@media screen and (nth($query, 1): nth($query, 2)) {
$default-grid-columns: $grid-columns;
$grid-columns: nth($query, 3);
@content;
$grid-columns: $default-grid-columns;
}
}
@else if length($query) == 4 {
@media screen and (nth($query, 1): nth($query, 2)) and (nth($query, 3): nth($query, 4)) {
$default-grid-columns: $grid-columns;
$grid-columns: $total-columns;
@content;
$grid-columns: $default-grid-columns;
}
}
@else if length($query) == 5 {
@media screen and (nth($query, 1): nth($query, 2)) and (nth($query, 3): nth($query, 4)) {
$default-grid-columns: $grid-columns;
$grid-columns: nth($query, 5);
@content;
$grid-columns: $default-grid-columns;
}
}
@else {
@warn "Wrong number of arguments for breakpoint(). Read the documentation for more details.";
}
}
@mixin outer-container {
@include clearfix;
max-width: $max-width;
margin: {
left: auto;
right: auto;
}
}
@mixin span-columns($span: $columns of $container-columns, $display: block) {
$columns: nth($span, 1);
$container-columns: container-span($span);
// Set nesting context (used by shift())
$parent-columns: get-parent-columns($container-columns);
$direction: get-direction($layout-direction, $default-layout-direction);
$opposite-direction: get-opposite-direction($direction);
$display-table: is-display-table($container-display-table, $display);
@if $display-table {
$padding: get-padding-for-table-layout($columns, $container-columns);
display: table-cell;
padding-#{$direction}: $padding;
width: flex-grid($columns, $container-columns) + $padding;
} @else {
display: block;
float: #{$opposite-direction};
@if $display == collapse {
width: flex-grid($columns, $container-columns) + flex-gutter($container-columns);
&:last-child {
width: flex-grid($columns, $container-columns);
}
} @else {
margin-#{$direction}: flex-gutter($container-columns);
width: flex-grid($columns, $container-columns);
&:last-child {
margin-#{$direction}: 0;
}
}
}
}
// Site-wide base styles.
// Setting root sizes and base styles.
html {
-webkit-text-size-adjust: 100%;
-ms-text-size-adjust: 100%;
@include rootsize;
}
body {
background-color: $color-bg;
color: $color-body;
overflow-x: hidden;
font-family: unquote(map-get($bodytype, font-family));
font-style: normal;
font-weight: map-get($bodytype, regular);
line-height: 2rem;
@include fontsize(zeta, all);
}
main {
margin: 20px auto;
padding: 0 20px;
@include outer-container;
@include media($xxs) {
padding: 0;
margin-top: 10px;
}
}
.box-model {
@include media($sm) {
@include span-columns(6);
}
@include media($md) {
@include span-columns(7);
}
#content, #generated-size {
@include media($lg) {
@include span-columns(4 of 7);
}
}
#generated-size {
@include media($lg) {
@include span-columns(3 of 7);
@include omega;
}
}
}
.controls {
@include media($sm) {
@include span-columns(6);
}
@include media($md) {
@include span-columns(5);
}
}
/* ************************************************************ */
// BOX Style mixins
// --------------------------------------
@mixin box-property-labels($position) {
.box-property-vertical {
left: $position;
}
.box-property-horizontal {
top: $position;
}
}
@mixin stagger-labels($position) {
.box-property-vertical {
left: $position;
}
.box-property-horizontal {
top: $position;
}
}
%hover-opacity {
opacity: 1;
}
@mixin box-styles($color, $border-style, $position-outer:false, $position:0) {
background: $color;
border: 1px $border-style lighten($color-body, 45%);
text-shadow: 0px 1px 1px transparentize(lighten($color, 20%), 0.3);
span::before,
span::after {
color: darken($color, 40%);
}
&:hover {
background-color: $color !important;
@if position-outer {
.box-property-vertical,
.box-property-horizontal {
@extend %hover-opacity;
opacity: 1;
}
}
}
@if $position-outer {
@include stagger-labels($position);
}
}
/*****************************************
* CONTROL STYLES
******************************************/
// Fieldset/Legend
// ---------------------------------------
fieldset {
border: 1px solid $color-dark-light;
padding: 0 10px 5px;
}
#generated-size fieldset {
min-height: 114px;
}
legend {
text-transform: uppercase;
font-weight: 300;
font-size: 1.3em;
}
@each $property, $color in $boxes {
##{$property} {
legend {
color: $color;
}
}
}
label, .value, .toggle-text {
font-size: 60%;
display: inline-block;
white-space: nowrap;
font-family: unquote(map-get($bodytype, font-family));
letter-spacing: 1.5px;
}
.control-set {
margin-top: 20px;
width: 100%;
label, .toggle-text {
width: 25%;
text-transform: uppercase;
}
.slider {
width: 65%;
@include media($md) {
width: 55%;
}
@include media($lg) {
width: 63%;
}
}
.value {
width: 7%;
text-align: right;
}
}
.box-model #content label {
width: 25%;
@include media($md) {
width: 22%;
}
}
#box-sizing label {
width: 40%;
}
// Radio Button
// ---------------------------------------
.radio {
position: relative;
margin: 0 1rem 0 0;
cursor: pointer;
&::before,
&::after {
@include transition(all 0.3s ease-in-out);
content: "";
position: absolute;
top: -0.2rem;
left: -0.2rem;
z-index: 1;
width: $knob-size;
height: $knob-size;
background: $color-dark-light !important;
border-radius: 50%;
}
&:checked,
&.ng-valid-parse {
&::before {
@include transform(scale(0, 0));
border-color: $color-box-sizing !important;
border-width: $knob-border-width !important;
}
&::after {
border: $knob-border-width solid $color-box-sizing !important;
background: $color-bg !important;
}
}
}
// Range Sliders
// ---------------------------------------
slider, [slider] {
display: inline-block;
position: relative;
height: 7px;
width: 63%;
vertical-align: middle;
margin: 5px;
div {
white-space: nowrap;
position: absolute;
&.handle {
border: $knob-border-width solid;
cursor: pointer;
width: $knob-size;
height: $knob-size;
top: -8px;
background-color: $color-bg;
z-index: 2;
border-radius: 100%;
&::after {
content: '';
width: 8px;
height: 8px;
position: absolute;
left: 6px;
border-radius: 100%;
background-color: transparent;
}
&.active::after {
background-color: transparent;
}
}
&.bar {
width: 100%;
height: 100%;
border-radius: 7px;
background: $color-dark-lighter;
overflow: hidden;
.selection {
width: 0;
height: 100%;
}
}
// TODO: remove Bubble functionality from slider directive
&.bubble {
display: none;
cursor: default;
top: -22px;
padding: 1px 3px;
font-size: 0.7em;
&.active {
display: inline-block;
}
&.limit {
color: $color-dark-lightest;
}
}
}
}
.bubble.value.low.ng-binding.active {
display: none;
}
.bar-color {
height: 20px;
border-right: 3px solid;
box-sizing: content-box;
}
@each $property in $sliders {
##{$property} {
.slider-selection, .bar-color {
background-color: map-get($boxes, #{$property});
}
.bar-color,
.handle {
border-color: map-get($boxes, #{$property});
}
}
}
@media (min-width: 951px) and (max-width: 1244px) {
.controls {
.sliders label {
display: block;
line-height: 1rem;
&:first-of-type {
margin-top: 8px;
}
}
slider, [slider] {
width: 87%;
}
}
}
// Toggle
// ---------------------------------------
.toggle {
display: none;
&, &::after, &::before, & *, & *::after, & *::before, & + .toggle-control {
box-sizing: border-box;
&::selection {
background: none;
}
}
+ .toggle-control {
outline: 0;
top: 10px;
margin-bottom: 8px;
width: 52px;
position: relative;
cursor: pointer;
user-select: none;
padding: 3px;
@include transition(all 0.3s ease-in-out);
background: $color-bg;
border: $knob-border-width/2 solid $color-dark-lighter;
border-radius: 2em;
&::after, &::before {
position: relative;
display: block;
content:"";
width: $knob-size;
height: $knob-size;
}
&::after {
left: 0;
@include transition(all 0.3s ease-in-out);
background: $color-dark-light;
border-radius: 50%;
}
&::before {
display: none;
}
}
&:checked {
+ .toggle-control {
border: $knob-border-width/2 solid $color-dark-lighter;
&::after {
left: 50%;
height: $knob-size;
width: $knob-size;
background: $color-bg;
border: $knob-border-width solid $color-content;
}
}
}
}
.toggle-text {
vertical-align: text-bottom;
margin-left: 5px;
}
// Generated Size Change Animation (ng-animate)
// ---------------------------------------
.generated-direction {
display: inline-block;
width: 150px;
}
.generated-width,
.generated-height {
font-size: 60%;
white-space: nowrap;
font-family: unquote(map-get($bodytype, font-family));
letter-spacing: 1.5px;
text-transform: uppercase;
line-height: 3rem;
@include media($sm) {
line-height: 2rem;
}
.changes {
text-transform: none;
display: inline-block;
line-height: 25px;
padding: 0 5px;
background: #2F353E;
border-radius: 3px;
color: $color-content;
@include transition(color 0.4s ease-in-out, background 0.4s ease-in-out);
&[class*="-add"] {
color: darken($color-content, 45%);
background: $color-content;
}
&.highlight {
&.ng-enter {
background: #2F353E;
color: $color-content;
&.ng-enter-active {
color: darken($color-content, 45%);
background: $color-content;
}
}
&.ng-leave {
color: darken($color-content, 45%);
background: $color-content;
&.ng-leave-active {
background: #2F353E;
color: $color-content;
}
}
}
}
}
/*****************************************
* DIAGRAM STYLES
******************************************/
#diagram {
margin-left: 5px;
clear: left;
padding-top: $gutter;
}
.box {
position: relative;
&:hover {
.box-property {
background: $color-white;
}
.box-property-vertical,
.box-property-horizontal {
opacity: 0;
}
}
}
.box-property {
font-family: unquote(map-get($monospacetype, font-family));
@include transition(width 0.3s linear, height 0.3s linear);
position: absolute;
}
.box-padding {
@include box-styles($color-padding, dashed, true, 40%);
}
.box-border {
@include box-styles($color-border, solid, true, 50%);
}
.box-margin {
@include box-styles($color-margin, dashed, true, 60%);
/** To Decode this SVG image, paste the css here: www.svgeneration.com/tools/base-64-decoder */
background-color: #de6a63;
background-image:url('');
}
.box-inner {
@include box-styles($color-content, solid);
text-align: center;
&::before {
color: darken($color-content, 35%);
content: attr(data-width) " x " attr(data-height);
position: absolute;
left: 0;
top: 50%;
margin-top: .5em;
width: 100%;
font-size: .75em;
white-space: nowrap;
}
}
// Box-Property Main Labels
// ---------------------------------------
.property-label {
font-family: unquote(map-get($headingtype, font-family));
text-transform: uppercase;
font-weight: 400;
letter-spacing: 2px;
font-size: 11px;
top: -6px;
@include media($sm) {
top: -12px;
}
left: 5px;
position: relative;
&#property-label-padding {
color: darken($color-padding, 15%);
}
&#property-label-border {
color: darken($color-border, 28%);
}
&#property-label-margin {
color: darken($color-margin, 20%);
}
&#property-label-content {
color: darken($color-content, 20%);
float: left;
top: -6px;
@include media($sm) {
top: -8px;
}
}
}
#padding-v::before {
top: -8px;
}
%box-property-position {
position: absolute;
top: 0;
left: 0;
z-index: 2000;
&::before, &::after {
position: absolute;
font-size: 0.75em;
text-align: center;
}
}
%property-vertical {
left: -0.5em;
width: 100%;
}
%property-horizontal {
margin-top: -0.65em;
width: 2em;
height: 100%;
}
$properties: zip(vertical horizontal, height width, top left, bottom right, -0.5em -1em);
@each $direction, $dimension, $offset-before, $offset-after, $amount in $properties {
.box-property-#{$direction} {
@extend %box-property-position;
#{$dimension}: 100%;
&::before {
content:attr(data-#{$offset-before});
@extend %property-#{$direction};
#{$offset-before}: $amount;
}
&::after {
content: attr(data-#{$offset-after});
@extend %property-#{$direction};
#{$offset-after}: $amount;
}
}
}
<link href="https://fonts.googleapis.com/css?family=Source+Sans+Pro:300,400|Source+Code+Pro:300,400" rel="stylesheet" />
<link href="https://codepen.io/carolineartz/pen/MYxjEv" rel="stylesheet" />
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment