I have created a new uploader. The instructions to use it can be found on this blog.
Gui from the future speaking. For any of the exploits to work below, you MUST be on the dsipaint.com version of my profile. Due to HullBreach's ban on CORS requests to DSi Paint (which is a good security practice), the exploits will not work on 3DS Paint.
EDIT: I decided to hide this portion of the blog since it isn't as useful anymore. Read at your own risk.
The easy way out: draw one, dummy. Then choose it in your profile settings.
Those of you who have keen eyes may note that my profile picture has transparency. Those of you who are worth your salt will know that you can't draw transparency on DSi Paint. Good luck finding that glass crayon.
So how did I do it?
The answer is surprisingly simple.
I sent a request to the server that asked it to save an image to its server. Then I can use that image as my profile picture.
Now how can you do it?
The secret ingredient is crime JS. And some Java. The two languages which I am the most familiar with. (HTML doesn't count.)
The first step is to convert the image into a format known as Base 64. For context, it looks something like this:
and so on for thousands of characters. It essentially encodes an image into a big, messy string. But why Base 64? I'll get to that in a bit.
Doing this in Java isn't that hard. I followed Baeldung's guide on converting images to Base 64. Once the program converted the image to Base 64, it wrote the data to a file for safekeeping. Here's the code that I used. My code does have dependencies, but handling dependencies is way, waaay outside of the scope of this blog.
(I did this in Java since that's the language I'm most familiar with. However, you could probably do the same thing with C++ or Kotlin or, god forbid, Rust. You can't use JS and HTML here since HTML canvases can't store transparency data.)
Alright, so you have the data of the image in one massive string. Now what? Now we send it over.
Sending it over was the harder part to figure out for me, but, thankfully, the code of DSi Paint is well-formatted and concise (despite its quality), so once I knew where to look it wasn't hard to find out. (Please note I said DSi Paint, not 3DS Paint. I'm not sure if this would work on 3DS Paint, but I'm optimistic that it would.)
It turns out that, like most of the rest of DSi Paint, the painting app operates on XMLHttpRequests. These requests allow the client to send data to the server without refreshing the page, making it a very powerful tool. Sending images over an XMLHttpRequest is hard, however. There's multiple ways to do it, but converting the image into a string of characters is likely the simplest way to do it since transmitting a message full of letters and numbers is trivial. Clever.
Cool, but how do you go about sending these XMLHttpRequests? You need to open the developer console on a major modern web browser.
First navigate to DSi Paint's DSi Paint. Then paste this script into the dev console. On Chrome, you can easily open this by hitting "Ctrl + Shift + I" and then navigating to the "Console" tab. Of course, replace the first string with your Base 64 image data. Once the script has been run (by pressing Enter) and the bottom reads "Image saved to gallery.", you're set.
Now cruise over to your user settings and select that image as your new profile picture. What's cool is that the image is saved as a legitimate DSi Paint image, so it saves there forever. Unless you delete it. But why would you ever do that?
Also, this blog is dedicated to banjo, who guessed first try who my profile picture is. Absolute madlad. Shoutouts to Pop and CM as well since they guessed in the right neck of the woods.
Have fun making custom profile pictures!
NOTE TO HULLBREACH:
On behalf of the entire community, please don't patch this. Not that it breaks anything. But if you do, I want to figure out how since I can't think of how you would do it.
UPDATE: Since some of you might not be able to follow the above directions, I decided to create a tool that does this for you. To use this tool, go to the DSi Paint application and save a random painting. It doesn't matter what since it's going to be overwritten with your upload data. You must do this before each upload to prevent previous images from being overwritten.
Once you do that, navigate to my profile page and scroll to the bottom of my profile bio. What's important is test #6. First click the button above it, then the test #6 button itself. Two new things should pop up below the buttons. From there, choose the file you want to use and then upload it by clicking the button below. A load of rubbish should appear on your screen, and click past it. Once it confirms that your image has been saved, you're set! Now you can use that image as your profile picture.
UPDATE: I created an alternate version of the tool for those of you who want to upload images but can't get a hold of a web browser that supports file uploads. To use this tool, follow the same steps for uploading the image except for clicking on "Include jQuery." That part isn't needed since I made this in vanilla JS this time around. Also, instead of uploading an image, just type in the web address of the image you want to upload and the server will take care of everything for you. Again, you'll see a whole load of gibberish and then hopefully a message that confirms that you've uploaded the image successfully. Once that happens, you're all set!
I'm doing this because some of y'all don't have access to modern devices. Lookin' at you, Draconid Jo. EDIT: So it looks like the DSi doesn't support CORS, so that means any exploits above will not work. jQuery also can't be loaded dynamically, so that pathway doesn't work either. That means that the DSi can't upload pfp's. Sorry. I don't know about the 3DS or 2DS, though.
UPDATE: I wasn't really clear about how to do it via URL, sorry. That would be test #7, and there's no need to click on the "Include jQuery" button.
Thanks to a tad bit of testing, I now know that CORS is possible on an old 2DS, but since I don't have one for myself, I can't make a viable version for those devices. Sorry about that.
I can't do this myself RN (for a wide array of different reasons), but it was still an interesting read, and it will no doubt help many people to do the same thing you did. (I'll probably use this trick myself some day!)
GuiedGui
16 Apr 2020 15:36
In reply to Draconid_Jo
You know what, you saying that gives me an idea. I could insert a tool into my profile that allows you to add a profile picture. This would make this significantly easier.
But I would have to figure out how to do that...
aidansocool
25 Apr 2020 19:17
In reply to GuiedGui
Can u do the same plz.
GuiedGui
25 Apr 2020 19:53
In reply to aidansocool
What's the matter? I don't think I understand.
aidansocool
25 Apr 2020 19:55
In reply to GuiedGui
Okay.
Draconid_Jo
18 Apr 2020 13:04
In reply to GuiedGui
Really?
Thanks, I'd really appreciate that!
I'd probably do it to my current PFP, actually.
(I'd prefer a clear background over the black one it has RN.)
GuiedGui
18 Apr 2020 15:23
In reply to Draconid_Jo
It's all done! :3 Just read the update
Draconid_Jo
19 Apr 2020 18:22
In reply to GuiedGui
Cool, I'll check it out once I'm finished checking my Notifications!
Thanks! (I just hope I can use it on my 2DS…
GuiedGui
19 Apr 2020 18:24
In reply to Draconid_Jo
Oh, it might now work on a 2DS since you need to be able to upload images. Should I make a similar tool that allows you to insert a image URL?
Draconid_Jo
19 Apr 2020 19:28
In reply to GuiedGui