SVG view box crop using jquery - javascript
I am doing sticker website for my client. They need image crop within given SVG image. That SVG image is unshaped like Halloween face (see below).
When i upload an image, it will display within the given image. The rest of the image will hide.
How can i achieve this?
This is really quite easy with SVG clipPaths. It has good browser support as well, except of course IE.
.defs {
position: absolute;
width: 0;
height: 0;
}
.demo {
text-align:center;
}
.star image {
clip-path: url(#star-clip);
}
.hex image {
clip-path: url(#hex-clip);
}
.heart image {
clip-path: url(#heart-clip);
}
.shape image {
clip-path: url(#shape-clip);
}
<svg class="defs">
<defs>
<clipPath id="star-clip">
<polygon id="star" points="150,7.3 196.4,101.3 300,116.3 225,189.4 242.7,292.7 150,243.9 57.3,292.7 75,189.4 0,116.3 103.6,101.3" />
</clipPath>
<clipPath id="hex-clip">
<polygon id="hex" points="222.5,295 77.5,295 5,150 77.5,5 222.5,5 295,150" />
</clipPath>
<clipPath id="heart-clip">
<path id="heart" d="M248.078,5.883c-36.691-14.739-77.771-0.839-98.517,31.125C128.817,5.044,87.735-8.856,51.043,5.883 C9.354,22.632-10.863,70.009,5.887,111.696c16.06,39.98,143.314,139.607,143.314,139.607l0.359,0.28l0.36-0.28 c0,0,127.251-99.627,143.314-139.607C309.986,70.009,289.768,22.632,248.078,5.883z"/>
</clipPath>
<clipPath id="shape-clip">
<path id="shape" d="M132.9,0.9c-12.5,1-34.7,4.4-44.9,6.7c-33.5,7.6-55.7,28.2-63.8,58.8c-1.8,6.9-2.2,9.7-2.3,20.1c0,10.5,0.2,13.3,2.3,21.6 c9.5,40.7,33.2,90.2,71.5,149.8c10.5,16.4,14.8,21.8,22.8,29.3c4.3,3.9,8.3,6.8,13,9.2l6.9,3.4h10.9c10.9,0,10.9,0,17-2.9 c7.6-3.7,14.3-8.7,21.4-16.2c6.9-7.2,10.5-12.3,21.9-30.3C248,188.7,272,134.3,277.4,96.6c1.2-8.6,0.5-22.4-1.7-31.3 c-5.6-23.4-20.8-41.2-43.6-50.9c-15.3-6.5-27.5-10-44.5-12.5C177.7,0.5,144.9-0.1,132.9,0.9z"/>
</clipPath>
</defs>
</svg>
<div class="demo heart">
<svg width="300" height="300">
<image xlink:href="http://lorempixel.com/output/city-q-c-300-300-4.jpg" width="300" height="300" />
</svg>
</div>
<div class="demo star">
<svg width="300" height="300">
<image xlink:href="http://lorempixel.com/output/cats-q-c-300-300-10.jpg" width="300" height="300" />
</svg>
</div>
<div class="demo hex">
<svg width="300" height="300">
<image xlink:href="http://lorempixel.com/output/people-q-c-300-300-8.jpg" width="300" height="300" />
</svg>
</div>
<div class="demo shape">
<svg width="300" height="300">
<image xlink:href="http://lorempixel.com/output/nightlife-q-c-300-300-6.jpg" width="300" height="300" />
</svg>
</div>
You'll need to define your clipPath and then apply it to your image. In the demo, I created a separate SVG element that contains all of the shapes as path or polygon definitions and simply applied them using external styles. You can of course do this in a single SVG, such as:
<svg width="120" height="120">
<defs>
<clipPath id="shape-clip">
<path id="shape" d="M60,117.4c-20.4,0-44.8-53.6-49-76.4C7.3,21,26.6,4.2,46.9,2.6c7-0.5,19.8-0.2,26.2,0 C93.5,3.4,112.7,21,109,41C104.8,63.8,79.3,117.4,60,117.4z"/>
</clipPath>
</defs>
<image xlink:href="http://lorempixel.com/output/nightlife-q-c-120-120-6.jpg" width="120" height="120" style="clip-path: url(#shape-clip);" />
</svg>
Some things to keep in mind:
Your defs need to be defined in the source order before they are used (so you can't put them at the end of your page for example).
The SVG element defs will not display, but they will take up space in the rendered page. I just force them out of the document flow with position: absolute and give them a height and width of 0 to make sure they don't mess with my layout.
For more information, you might want to look at these:
Clipping and Masking in CSS, CSS Tricks
SVG Clip Path Tutorial, Jenkov.com
Clipping, Masking and Compositing Specification, W3C
Related
Svg icons disappear when I resize window
I have an SVG icon like this: <svg width="19" height="19" viewBox="0 0 19 19" fill="none" xmlns="http://www.w3.org/2000/svg" xmlnsXlink="http://www.w3.org/1999/xlink"> <rect width="19" height="19" fill="url(#pattern0)"/> <defs> <pattern id="pattern0" patternContentUnits="objectBoundingBox" width="1" height="1"> <use xlinkHref="#image0" transform="scale(0.00444444)"/> </pattern> <image id="image0" width="225" height="225" xlinkHref=""/> </defs> </svg> It displays on screen width > 1279 but disappears from the screen width <= 1279. Because of the page's layout, I have three divs that render on different screen sizes and contain this svg. I tried using svgs with different ids but the same content. However, it didn't work for me. Can somebody help me with that? logic code: component that uses svg icons (all of them disappear on width <= 1279 except star svgs that rendered in a list) <div className="position__reviews--social-wrapper"> <div className="position__reviews--social"> <Icon className="position__reviews--social-icon" name="google" /> <Icon className="position__reviews--social-icon" name="vivino" /> </div> <div className="position__reviews--vivino-wrapper"> <Icon className="position__reviews--vivino-text-icon" name="vivino-text" /> {[...Array(5)].map((star,idx) => idx + 1 <= redStars ? <Icon className="position__reviews--star-red" name="position__review-star-red" /> : <Icon className="position__reviews--star-gray" name="position__review-star-gray" />)} </div> </div> Icon (svgIcons is a list of jsx items: () => (<svg>...</svg>) ): const Icon = ({ name, fill, onClick, className }) => { const svg = svgIcons[name]; const iconClasses = cn(className, { 'icon--active': !!onClick, }); return svg ? ( <span className={iconClasses} onClick={onClick}> {svg(fill)} </span> ) : null; }; css: .position__reviews--social-icon { border: 1px solid #C4C4C4; box-sizing: border-box; border-radius: 3px; margin-right: 6px; padding: 4px; padding-top: 9px; padding-bottom: 1px; cursor: pointer; } What happens: width > 1279: width <= 1279:
This: <svg width="19" height="19" viewBox="0 0 19 19" fill="none" xmlns="http://www.w3.org/2000/svg" xmlnsXlink="http://www.w3.org/1999/xlink"> <rect width="19" height="19" fill="url(#pattern0)"/> <defs> <pattern id="pattern0" patternContentUnits="objectBoundingBox" width="1" height="1"> <use xlinkHref="#image0" transform="scale(0.00444444)"/> </pattern> <image id="image0" width="225" height="225" xlinkHref=""/> </defs> </svg> was most likely created by a developer in a K-Hole "Falling into a k-hole" is slang for how it feels when you take a high enough dose of ketamine that your awareness of the world around you and your control over your own body become so profoundly impaired that you're temporarily unable to interact with others—or the world around you. It wraps a Base64 data (the Google G logo) in an <image> in a <pattern> in a <rect> in a <svg> I would ask that developer what the meaning is, because all you have to do is: wrap the Base64 data in an IMG element <img width="19" src=""> If you do want the contrived SVG, write proper SVG: <svg width="19" height="19" viewBox="0 0 19 19"> <rect width="19" height="19" fill="url(#pattern0)"/> <defs> <pattern id="pattern0" patternContentUnits="objectBoundingBox" width="1" height="1"> <use href="#image0" transform="scale(0.00444444)"/> </pattern> <image id="image0" width="225" height="225" href=""/> </defs> </svg>
Making inline SVG responsive in browsers without using javascript?
I've been reviewing articles like this one which are really good, but possibly dated, and I'm wondering if any progress has been made on a simple way to make inline svg responsive while maintaining the aspect ratio? In other words if we have svg with a 1:1 aspect ratio inside a div container, and the container shrinks from 400x400px to 200x200 px, then the view box width and height parameters double.
I am confused about what you want. I don't know what you mean, in your question, when you talk about "view box width and height parameters double". Perhaps you are confused about how viewBox works. As long as the SVG has a viewBox it should be responsive. In the following example, I've put the SVG in a div container and animated the container size to simulate the page size changing. #container { width: 200px; background: linen; animation: scale 1s alternate infinite; } #keyframes scale { from { width: 200px; } to { width: 100px; } } <div id="container"> <svg viewBox="0 0 100 100"> <circle cx="50" cy="50" r="40" fill="gold"/> <rect x="10" y="50" width="20" height="50" fill="gold"/> <rect x="70" y="50" width="20" height="50" fill="gold"/> <circle cx="35" cy="45" r="5"/> <circle cx="65" cy="45" r="5"/> </svg> </div> But it could just as easily be a container that is the width of the page. Try running the following snippet. Then click "Fullpage" and resize the browser window.: #container { width: 100%; background: linen; } <div id="container"> <svg viewBox="0 0 100 100"> <circle cx="50" cy="50" r="40" fill="gold"/> <rect x="10" y="50" width="20" height="50" fill="gold"/> <rect x="70" y="50" width="20" height="50" fill="gold"/> <circle cx="35" cy="45" r="5"/> <circle cx="65" cy="45" r="5"/> </svg> </div>
If you don't need to change svg with css or modify it, I suggest you to use it as an img element. You create an img and put the path to your svg in the src. Here's what I mean: https://codepen.io/tlemaitre/pen/PBKGOo
I've recently had a similar problem, I have a container that needs to be preserveAspectRatio="none" to keep the shape fluid on different window sizes but the image inside gets gets distorted. I've not seen any solution to this problem with or without JS, so I don't think it's possible.
ScrollMagic, reverse z-index order
I'm using scrollmagic to tween svg clippaths as you scroll. Usually with svgs(or divs or sections or whatever) the last thing you specify in your html is on top in terms of z-index, but you change that in css. I tried to do this with my svgs, so that while the first svg tweens, the other one scrolls up behind it. It seems like scrollmagic is preventing my z-indexing from working though. Any ideas? http://codepen.io/kathryncrawford/pen/BoXOMJ <div id="scene"> <svg id="svg1" height="500" width="800"> <image id="img1" xlink:href="http://placecage.com/800/500" x="0" y="0" width="800" height="500"/> <defs> <clipPath id="clip1"> <circle id="circle1" stroke="#000000" stroke-miterlimit="10" cx="400" cy="300" r="300" /> </clipPath> </defs> </svg> </div> <div id="scene2"> <svg id="svg2" height="500" width="800"> <image id="img2" xlink:href="http://fillmurray.com/800/500" x="0" y="0" width="800" height="500"/> <defs> <clipPath id="clip2"> <circle id="circle2" stroke="#000000" stroke-miterlimit="10" cx="400" cy="300" r="300" /> </clipPath> </defs> </svg> </div> CSS #img1 { clip-path: url(#clip1); } #img2 { clip-path: url(#clip2); } #svg1, #circle1{ z-index: 2; } #svg2, #circle2{ z-index: 1; }
Crap, I literally just figured it out after posting this. I changed the z-index css declaration to the z-index on the scene divs that wrap the svgs. That worked. http://codepen.io/kathryncrawford/pen/BoXOMJ #scene{ z-index: 2; } #scene2{ z-index: 1; }
Change opacity of an area based on mouse position
Is it possible to change the opacity of background but only underneath the cursor area (for example a white small circle)? I am thinking of it a bit like a basic heatmap but the heat doesn't stay - it just follows the cursor. At the moment I have the following HTML: html { background-color: #000; width: 100%; height: 100%; } JS: $(document).mousemove(function(event){ var i = event.pageX.toPrecision(1) / 1000; $("html").css('opacity', i) }); Sorry this is probably a very basic starting point. Would I need to use canvas?
You can do that using svg What i did :- I placed two same images with same co ordinates,height and width and gave a circular clip-path to the one on top (which has full opacity) when mouse moves the position of the circle also changes $('svg').on('mousemove',function(e){ $('.a').attr('cx',e.pageX).attr('cy',e.pageY) }) .one{ opacity:.5; } <script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script> <svg width="500" height="500"> <clippath id="clip" > <circle cx="50" cy="50" r="50" class="a"/> </clippath> <image xlink:href="https://images.unsplash.com/photo-1474575981580-1ec7944df3b2?dpr=1&auto=format&fit=crop&w=1500&h=934&q=80&cs=tinysrgb&crop=&bg=" width="500" height="500" x="0" y="0" class="one"/> <image xlink:href="https://images.unsplash.com/photo-1474575981580-1ec7944df3b2?dpr=1&auto=format&fit=crop&w=1500&h=934&q=80&cs=tinysrgb&crop=&bg=" width="500" height="500" x="0" y="0" clip-path="url(#clip)"/> </svg>
Can't change SVG rect properties in three containers
After several tries, and code changes, I am unable to make rectangle inside SVG to change his position - don't even ask for animating. Obviously using jQuery SVG plugin plus animation extension. The problem: A SVGs wrapped inside three <div>, an inside y have a rectangle that need to be at y:0 after the document loads. And this is the code: var rect = jQuery('div.post-image').children('svg').svg().svg('get'); jQuery(rect).each(function(){ jQuery(this).change('.b1', {y:0}); }); Well, nothing happens with the rectangle, it keeps the original coordinate. Chrome console doesn't says anything either. Added: the HTML in question <a href="#" class="post-image" title="title"> <svg xmlns="http://www.w3.org/2000/svg" version="1.1" width="250" height="140" viewBox="0,0,250,140" overflow="hidden"> <switch> <g> <defs> <filter id="innershadow"> <feOffset dx="0" dy="0"></feOffset> <feGaussianBlur stdDeviation="7" result="offset-blur"></feGaussianBlur> <feComposite operator="out" in="SourceGraphic" in2="offset-blur" result="inverse"></feComposite> <feFlood flood-color="#000" flood-opacity="0.3" result="color"></feFlood> <feComposite operator="in" in="color" in2="inverse" result="shadow"></feComposite> <feComposite operator="over" in="shadow" in2="SourceGraphic"></feComposite> </filter> <pattern xmlns="http://www.w3.org/2000/svg" id="image-771" patternUnits="userSpaceOnUse" width="250" height="202"> <image href="example-310x250.jpg" xlink="http://www.w3.org/1999/xlink" x="0" y="0" width="250" height="202"></image> </pattern> <clipPath id="clip"> <polygon points="0,0 235,0 250,70 235,140 0,140 15,70"></polygon> </clipPath> </defs> <polygon points="0,0 235,0 250,70 235,140 0,140 15,70" style="fill: url(#image-771); filter:url(#innershadow);"></polygon> <rect class="b1" width="100%" height="100%" style="fill:rgb(0,92,148); opacity: 0.9;" clip-path="url(#clip)" x="0" y="98"></rect> <rect class="b2" width="60" height="25" style="fill:rgb(0,92,148); opacity: 0.9;" clip-path="url(#clip)" x="190" y="0"></rect> <rect class="b3" width="100" height="25" style="fill:rgb(0,0,0); opacity: 0.75;" clip-path="url(#clip)" x="0" y="0"></rect> </g> <foreignObject width="250" height="140"> <img width="250" height="125" src="example-fallback.jpg" alt="example" title="title"> </foreignObject> </switch> </svg> </a> I'm willing to use <canvas> for this, but I don't know what are the outcomes.
Found out what wast the problem: var rect = jQuery('a.post-image').children('svg').find('.b2, .b3'); jQuery(rect).each(function(){ jQuery(this).attr('y','-25'); }); Done, and without a plugin. Okey, not the best way (find instead of most direct selector), but it will cut it.