Nial
Nial

Reputation: 441

Set background-url css property to a memory image in .net

Please read about my following situation. So we have a web application in .net. Because some customers wanted to pay for some custom layout, we first started to use themes. It was ok until there were more than 10. Because after that it was difficult to maintain and more of them wanted custom design, I had to design a solution so that the user could change only a few css attributes through a front-end page.

So when a user clicked a button a folder with its name and another css file was generated and he could change the attributes and save the new file. This file was dynamically loaded on the Page_Load on some basebage. This was ok, until more and more users payed for this and now we have tens of folders and expect hundreds. There is also the problem of rights, by default IIS doesn't have writing right to the application folder. Also the deployment process builds a new version by deleting the whole app folder and recreating one so we had to do a workaround and make a copy of this mega folder that holds the users css. And anyway, I don't like the idea of having hundreds of files not under source control.

And now, I have a new requierement. The users to be allowed to upload a custom image to replace one of our own. So i said it is time to move evrything to the database. For basic attributes I solved it.

I have a "mutant" aspx page:

<%@ Page Language="C#" ContentType="text/css" %>
.labelText
{
  color:<%= WebApplication1.UserProfile.LabelColor%>
}

So the UserProfile class gets it's data from a table. And then in the page_load of the basepage

HtmlLink link = new HtmlLink();
link.Attributes.Add("type", "text/css");
link.Attributes.Add("rel", "stylesheet");
link.Attributes.Add("href", "~/Styles/Override.aspx");
this.Header.Controls.Add(link);

It works ok, the problem the images are used like this:

.divHeader
{
   background: 15px 0 url("logo.gif")
}

That logo.gig will be uploaded by the user,but I would like to be saved in the database. However when reading it as a stream I can't do something like:

background: 15px 0 url(WebApplication1.UserProfile.Logo)

Is there any otehr way I can specify an object in a css class?If it isn't what is the best approach? I was thinking about keeping the image in the db and also as a file, something similar to what we have now. I realize that if it were for an image control it would have been a lot easier, but this is the design now, and remaking it is a little bit troublesome Thanks for reading this novel and for any oppinions on this.

Upvotes: 0

Views: 1836

Answers (1)

user191966
user191966

Reputation:

I hope you're saving the mime-type value of every image that you save to database (mime type is reported by the request during upload). If you don't, you'll have to look up what a mime type would be based on image-file extension or its content format. You'll need this value when images are requested (from within CSS files).

Implement an HTTP handler (.ashx) specifically for these images (new file > generic handler > fill up ProcessRequest).

Then, replace url(logo.gif) in your css files with something like this:

url(/GetImage.ashx?user=1234&image=logo);

Obviously, you can have the filename and query to be whatever you like/need. Maybe you have user info in session, so you don't have to include user id there. Maybe you'll want to implement this handler only for logo images, and if you have user info in session, you can simply say (css):

url(GetLogo.ashx);

Then, within that ashx handler, write code to get the image from the database (or where ever it is), and stream it to the browser. Something like this:

byte[] imageData = ... // get this from db
MemoryStream ms = new MemoryStream(imageData); // System.IO namespace
Response.ContentType = "image/png"; // remember the mime-type?
ms.WriteTo(context.Response.OutputStream); // context is current HttpContext

Alternatively, you can inject the whole image into the CSS, by encoding it to base-64 format. Note that the size of css file will grow for the size of the image, and you're practically combining multiple downloads into 1, which may be a good or a bad thing.

Example below (google's logo); paste this in CSS (notice the size of that thing).

Besides CSS, the data in this example ( ...) can be used as URL with IMG tags, or manipulated by JS.

.somediv
{
    width:275px;
    height:95px;
    background-image: url();
}

+1 if you like this :)

Upvotes: 1

Related Questions