fabricjs / fabric.js

Javascript Canvas Library, SVG-to-Canvas (& canvas-to-SVG) Parser
http://fabricjs.com
Other
29.02k stars 3.51k forks source link

support filters in to svg #1577

Open kidzzz opened 10 years ago

kidzzz commented 10 years ago

Hello,

I've using fabricjs for our own project. Works great!

I have an issue of converting canvas to SVG, particular with image filters. We use the Multiply filter, an example would be:

"filters":[{"type":"Multiply","color":"rgb(241, 93, 34)"}]

The current version of toSVG() doesn't support image filters in SVG format. We've looked around but have not been able to find a away to do this. We know where to modify the code but don't know how the format would look like. Any help is appreciated.

Thanks!

Want to back this issue? Post a bounty on it! We accept bounties via Bountysource.

kangax commented 10 years ago

/cc @asturur

asturur commented 10 years ago

taking a look.

asturur commented 10 years ago

@kidzzz @kangax Actually the situation is that the image class has 2 elements, original image and image with filters applied. When we export to SVG we export the image eventually already filtered, but not any filter information. What are you asking is to export the original untouched image and then for every filter class create a toSvg() method that would export the filter ( assigning it an ID and then linking the filter to the image ). This is certainly possible, but i cannot see any good thing for fabric.js from it. Svg output would be more complex and bigger, Import would be more complicated ( actually filters are not supported in import, look at svg 171, the yellow car ) Can you be more specific on what you would like to obtain? maybe i just misunderstood.

kangax commented 10 years ago

Creating toSVG() on each filter certainly sounds like the right way to do it; the Fabric way, when each class/object is responsible for its own rendering, import, export, etc. But it is quite a big effort, like @asturur says. We can do this in the future, but meanwhile, why doesn't your multiply filter work if SVG is supposed to be created with already modified image?

asturur commented 10 years ago

Also there should be a 1:1 correspondence between filters you can create in fabric.js and filters in svg spec. There is no evidence that every filter will be supported by svg spec. There will be a subset of filter exportable and other not maybe.

kidzzz commented 10 years ago

@kangax @asturur

I appreciate all the comments. Such a great project and creators!

"When we export to SVG we export the image eventually already filtered, but not any filter information" -- @asturur, are you saying the SVG should already have filters applied to the image? This is not what I am seeing.

Please let me explain my situation: A user uploads a clip-art into the canvas, then he changes the background of the image to a different color. We achieve this by using the Multiply filter, so the JSON looks like this (Please note the image url isn't valid but I attach the clipart so you guys can test it out):

{"objects":[{"type":"image","originX":"left","originY":"top","left":62,"top":60,"width":239,"height":298,"fill":"rgb(0,0,0)","stroke":null,"strokeWidth":1,"strokeDashArray":null,"strokeLineCap":"butt","strokeLineJoin":"miter","strokeMiterLimit":10,"scaleX":0.55,"scaleY":0.55,"angle":0,"flipX":false,"flipY":false,"opacity":1,"shadow":null,"visible":true,"clipTo":null,"backgroundColor":"","src":"http://xyz.com/clipart_sample.png","filters":[{"type":"Multiply","color":"rgb(252, 185, 19)"}],"crossOrigin":""}],"background":""}

When I export the JSON to SVG, here is what I get (Please ignore the image path, I made it relative):

<?xml version="1.0" encoding="UTF-8" standalone="no" ?><!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" width="244" height="362" xml:space="preserve"><desc>Created with Fabric.js 1.4.7</desc><defs></defs><g transform="translate(127.73 141.95) scale(0.55 0.55)"><image xlink:href="clipart_sample.png" style="stroke: none; stroke-width: 1; stroke-dasharray: ; stroke-linecap: butt; stroke-linejoin: miter; stroke-miterlimit: 10; fill: rgb(0,0,0); opacity: 1;" transform="translate(-119.5 -149)" width="239" height="298" preserveAspectRatio="none"></image></g></svg>

So when I open this SVG file in Illustrator or IE, the background color of the clipart is not saved. Looking at the SVG string, I don't see anything indicating the Multiply filter has been applied.

Hope I explain the problem well enough. Thanks a lot!

clipart_sample

kidzzz commented 10 years ago

Hey guys,

Just want to see if you guys have a chance to take a look at the problem.

Thanks!

kidzzz commented 10 years ago

Hello @kangax @asturur ,

Would be super cool if you guys can give me a some ideas on how to resolve this issue.

Thanks.

kangax commented 10 years ago

Image xlink:href is not supported at the moment, AFAIR, so we need to work on it. I'll get to it when I get time; right now still busy with work.

asturur commented 10 years ago

mmmm so strange. with a dataurl you should solve. i ll check as weel, but sunday evening, i m with family now. for sure fabric will not make new png for you, so dataurl should be the solution.

kidzzz commented 10 years ago

Okay guys. Thank you for the responses.

asturur commented 10 years ago

i have some spare time i wanted to dedicate to fabricjs tonight. is there in this issue all the material i need to check the problem? do i need something else other than code and image you posted? Il 26/ago/2014 21:34 "kidzzz" notifications@github.com ha scritto:

Okay guys. Thank you for the responses.

— Reply to this email directly or view it on GitHub https://github.com/kangax/fabric.js/issues/1577#issuecomment-53475731.

kidzzz commented 10 years ago

Yes, I think everything is in this thread. You should be able to use the JSON and image to test the current toSVG() function. I also included a sample SVG output as a reference.

Thank you!

asturur commented 10 years ago

So i loaded the PNG of the crocodile. Then i did :

oImg.filters.push(new fabric.Image.filters.Multiply({color:'rgb(241, 93, 34)'}));
      oImg.applyFilters();

Then i exported toSVG and i obtained

<?xml version="1.0" encoding="UTF-8" standalone="no" ?><!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" width="900" height="500" xml:space="preserve"><desc>Created with Fabric.js 1.4.11</desc><defs></defs><g transform="translate(341.5 267)">
<image xlink:href="" x="-119.5" y="-149" style="stroke: none; stroke-width: 1; stroke-dasharray: ; stroke-linecap: butt; stroke-linejoin: miter; stroke-miterlimit: 10; fill: rgb(0,0,0); fill-rule: source-over; opacity: 1;" width="239" height="298" preserveAspectRatio="none"></image>
</g>
</svg>

That to our eyes is

image

So, what is wrong? The png image has been exported to data url image, and that is the only way to do it untill filter exporting is possible. Are you running under nodejs?

kidzzz commented 10 years ago

@asturur - No, nothing is wrong. I am totally aware toDataUrl() should work with the filter. I tried and it did work. It was not until @kangax mentioned filter export was not yet supported, I then realized the whole thing.

What I am hoping to achieve is:

<image xlink:href="clipart_sample.png" filter_information />

Since I am not familiar with image filters and their formats, I have not been able to write my own function to do the job. Any help in this regards is greatly welcome.

And yes, I am running under nodejs.

Thanks again.