I found this: width/height after transform
and several others, but nothing is not quite what I\'m looking for. What I want is to scale something to 50% of it\'s siz
I finally figured out a simple solution and it is much similar than where I started from.
I will admit, it took many attempts to figure this out (over the last year or so) to finally figure it out. I was actually looking for an answer to this question. I have had problems with this before. Anyways, I started working this out again and I did find a simple solution.
Anyways, the trick was to use the margin bottom together with the CSS calc script.
Also, you must have to have absolute height declared for the the object (div). I dont think this will work with if the div can expand, otherwise, after many attempts and hours on this I finally have a very simple, clean, working solution.
So, for example, height = 300px (not 30%, or no height). I would even set overflow-y = hidden if you can. If you want the div to grow (not have an absolute height), I believe you will need JavaScript. I did not try it. Otherwise this is working correctly in all tests.
I am sure this will work nice with SASS or LESS since you could declare some of the repeated things using variables in the CSS. I have not tried that yet.
Some important comments on my solution:
I have also have set up a Codepen to test this with and for sharing.
https://codepen.io/cyansmiles/details/yvGaVP
.box-wrap {
display: block;
position: relative;
box-sizing: border-box;
width: 100%;
margin: auto;
text-align: center;
margin: 30px 10px;
padding: 40px;
background: yellow;
}
.box-wrap:before {
content: "";
display: block;
box-sizing: border-box;
position: absolute;
background: rgba(0, 0, 0, 0.5);
width: calc(100% - 80px);
height: calc(100% - 80px);
top: 40px;
left: 40px;
}
.box {
display: inline-block;
box-sizing: border-box;
position: relative;
transform-origin: 50% 0;
width: 300px;
height: 150px;
background: red;
}
.box.size-80 {
transform: scale(0.8);
margin: 0 auto calc(-150px * (1 - 0.8));
}
.box.size-70 {
transform: scale(0.7);
margin: 0 auto calc(-150px * (1 - 0.7));
}
.box.size-60 {
transform: scale(0.6);
margin: 0 auto calc(-150px * (1 - 0.6));
}
.box.size-50 {
transform: scale(0.5);
margin: 0 auto calc(-150px * (1 - 0.5));
}
//etc
<div class="box-wrap">
<div class="box">
<h1>Box at 100%</h1>
</div>
</div>
<div class="box-wrap">
<div class="box size-80">
<h1>Box at 80%</h1>
</div>
</div>
<div class="box-wrap">
<div class="box size-70">
<h1>Box at 70%</h1>
</div>
</div>
<div class="box-wrap">
<div class="box size-60">
<h1>Box at 60%</h1>
</div>
</div>
<div class="box-wrap">
<div class="box size-50">
<h1>Box at 50%</h1>
</div>
</div>
The problem I noticed is that when element scales, browser change its pixels ratio, not pixels amount. Element is smaller but it doesn't change its actual pixel size in DOM. Because of that I don't think that CSS-only solution exist.
I put scalable element into container which keeps the same pixel ratio as rest of elements. Using Java Script I simply change container's size. Everything is still based on CSS3 transform: scale. Maybe JS code could be simplier, but now it's all about the idea (just a proof of concept);) Fiddle with two examples: http://jsfiddle.net/qA5Tb/9/
HTML:
<div class="overall-scalable">
<div class="scalable" scalex='0.5' scaley='0.5'>
Nunc et nisi ante. Integer in blandit nisi. Nulla facilisi. Vestibulum vulputate sapien eget mauris elementum sollicitudin. Nullam id lobortis dolor. Nulla vitae nibh vitae sem volutpat pretium. Nunc et nisi ante. Integer in blandit nisi. Nulla facilisi. Vestibulum vulputate sapien eget mauris elementum sollicitudin. Nullam id lobortis dolor. Nulla vitae nibh vitae sem volutpat pretium.
</div>
</div>
CSS:
.overall-scalable {width: 350px; height: 150px; overflow: hidden; -webkit-transition: all 1s;}
.scalable {color: #666; width: 350px; height: 150px; -webkit-transform-origin: top left; -webkit-transition: all 1s;}
JS:
$('button').click(function() {
$('.scalable').each(function(){
rescale($(this));
})
});
function rescale(elem) {
var height = parseInt(elem.css('height'));
var width = parseInt(elem.css('width'));
var scalex = parseFloat(elem.attr('scalex'));
var scaley = parseFloat(elem.attr('scaley'));
if (!elem.hasClass('rescaled')){
var ratioX = scalex;
var ratioY = scaley;
}else{
var ratioX = 1;
var ratioY = 1;
}
elem.toggleClass('rescaled');
elem.css('-webkit-transform', 'scale('+ratioX +', '+ratioY+')');
elem.parent().css('width', parseInt(width*ratioX) + 'px');
elem.parent().css('height', parseInt(height*ratioY) + 'px');
}
I had to tweak my code a little from above.
It now works on all sides (not just up and down). I see a little edit, I think that is CSS issue (you could add another pixel to code to resolve this).
Here is the working Codepen link. Feel free to let me know if you use it. It felt good to finally figure this out.
https://codepen.io/cyansmiles/pen/bLOqZo
ps. I am forking my codepen to add a moveable CSS.
.box-wrap {
display: block;
position: relative;
box-sizing: border-box;
width: 100%;
margin: auto;
text-align: center;
margin: 30px 10px;
padding: 40px;
background: yellow;
}
.box-wrap:before {
content: "";
display: block;
box-sizing: border-box;
position: absolute;
background: rgba(0, 0, 0, 0.5);
width: calc(100% - 80px);
height: calc(100% - 80px);
top: 40px;
left: 40px;
}
.box {
display: inline-block;
box-sizing: border-box;
position: relative;
transform-origin: 50% 0;
width: 300px;
height: 150px;
background: red;
color: #fff;
}
.box.size-80 {
background: red;
transform: scale(0.8);
margin: 0 calc((-300px * (1 - 0.8)) / 2) calc(-150px * (1 - 0.8));
}
.box.size-70 {
background: blue;
transform: scale(0.7);
margin: 0 calc((-300px * (1 - 0.7)) / 2) calc(-150px * (1 - 0.7));
}
.box.size-60 {
background: green;
transform: scale(0.6);
margin: 0 calc((-300px * (1 - 0.6)) / 2) calc(-150px * (1 - 0.6));
}
.box.size-50 {
background: orange;
transform: scale(0.5);
margin: 0 calc((-300px * (1 - 0.5)) / 2) calc(-150px * (1 - 0.5));
}
//etc
<div class="box-wrap">
<h1>This is a bunch of boxes!</h1>
<div class="box size-50">
<h1>Box at 50%</h1>
</div>
<div class="box size-60">
<h1>Box at 60%</h1>
</div>
<div class="box size-80">
<h1>Box at 80%</h1>
</div>
<div class="box size-70">
<h1>Box at 70%</h1>
</div>
<div class="box size-50">
<h1>Box at 50%</h1>
</div>
<div class="box size-60">
<h1>Box at 60%</h1>
</div>
<div class="box size-50">
<h1>Box at 50%</h1>
</div>
<div class="box size-60">
<h1>Box at 60%</h1>
</div>
<div class="box size-80">
<h1>Box at 80%</h1>
</div>
<div class="box size-70">
<h1>Box at 70%</h1>
</div>
<div class="box size-50">
<h1>Box at 50%</h1>
</div>
<div class="box size-60">
<h1>Box at 60%</h1>
</div>
</div>
<div class="box-wrap">
<div class="box">
<h1>Box at 100%</h1>
</div>
</div>
<div class="box-wrap">
<div class="box size-80">
<h1>Box at 80%</h1>
</div>
</div>
<div class="box-wrap">
<div class="box size-70">
<h1>Box at 70%</h1>
</div>
</div>
<div class="box-wrap">
<div class="box size-60">
<h1>Box at 60%</h1>
</div>
</div>
<div class="box-wrap">
<div class="box size-50">
<h1>Box at 50%</h1>
</div>
</div>