This is a story about what happens when you give a software engineer, a mechanical engineer, and a toddler too much free time...
When I saw @cassidoo 's latest newsletter code challenge, it sparked my interest because it reminded me of old SAT problems I used to practice. Here's the problem:
You have to order wrapping paper for presents.
Given the length, width, and height of the boxes
you need to wrap,
return the number of square feet (or whatever units you want)
of wrapping paper you need to order.
Extra credit: allow for other shapes
of presents and their dimensions!
So, just finding the surface area of an orthotope (gift with rectangular or square sides) could look like this:
function getOrthotopeSurfaceArea(length, width, height) {
let firstVal = length * width * 2
let secondVal = length * height * 2
let thirdVal = height * width * 2
return firstVal + secondVal + thirdVal
}
And, as Cassidy suggests, we could make this more difficult by incorporating other shapes. I thought of spheres or Klein bottles.
But then, I thought about how this would work if I wanted to wrap a present in real life. How would I account for the overlap? The little bit of extra that you fold over the ends?
This is when I turned to my husband, who has a degree in mechanical engineering, so I guessed he'd be really good at making this challenge harder simplifying the problem.
He came up with this solution:
^ As you can see, our toddler helped him. She informed us that she had written an 'A, a O, a E'. We took note in case this information became useful later. It did not. But we are a team so the goal is respecting each other's input, even if it is not used in the final version of the solution.
Basically, there are these steps from the notes:
Re-order the length, width, and height from longest to shortest. (This would prove to be the hardest step for me to code.)
Find the overlap, or length and width tolerances. If the length or width was greater than 4 inches, the tolerance would be 2 inches, otherwise it would be half the length or width.
Based on these values, he wrote two equations to solve for the paper's width and length.
Length = longest entered value + 2(shortest value) + 2(length tolerance)
Width = shortest entered value + 2(middle value) + 2(width tolerance)
Now, the first thing I did was write the code for step 1, how to order three integer values according to magnitude. It was placeholder stuff -- a bunch of 'if... then' conditions that probably covered it, right? But then my husband came back into the room with all the possibilities listed:
Holy boolean expressions, batman! I did not want to type all that out.
So I ended up using a pointer approach for this step.
function arrangeByValue(l, w, h) {
let arrayToSort = [l, w, h]
for (let i = 0; i < arrayToSort.length - 1; i++) {
let firstPointer = arrayToSort[i]
let secondPointer = arrayToSort[i + 1]
if (secondPointer > firstPointer) {
let originalSmallerVal = arrayToSort[i]
arrayToSort[i] = arrayToSort[i + 1]
arrayToSort[i + 1] = originalSmallerVal
}
}
return arrayToSort
}
Now I could use that function in my code for steps 2 & 3!
Here's what I did:
function wrapThatGift(length, width, height) {
let arrangedArray = arrangeByValue(length, width, height)
let longest = arrangedArray[0]
let middle = arrangedArray[1]
let shortest = arrangedArray[2]
let widthTolerance
if (middle <= 4) {
widthTolerance = middle * 0.5
}
if (middle > 4) {
widthTolerance = 2
}
let lengthTolerance
if (length <= 4) {
lengthTolerance = length * 0.5
}
if (length > 4) {
lengthTolerance = 2
}
let paperWidth = shortest + 2 * middle + 2 * widthTolerance
let paperLength = longest + 2 * shortest + 2 * lengthTolerance
let wrappingPaperSurfaceArea = paperLength * paperWidth
return `${paperLength} inches by ${paperWidth} inches `
}
First, I re-arranged the values using the previous function:
let arrangedArray = arrangeByValue(length, width, height)
let longest = arrangedArray[0]
let middle = arrangedArray[1]
let shortest = arrangedArray[2]
Then, I found the tolerances:
let widthTolerance
if (middle <= 4) {
widthTolerance = middle * 0.5
}
if (middle > 4) {
widthTolerance = 2
}
let lengthTolerance
if (length <= 4) {
lengthTolerance = length * 0.5
}
if (length > 4) {
lengthTolerance = 2
}
Then, I translated the equations into code:
let paperLength = longest + 2 * shortest + 2 * lengthTolerance
let wrappingPaperSurfaceArea = paperLength * paperWidth
And returned the desired values:
return `${paperLength} inches by ${paperWidth} inches `
And voila!
When I ran node index.js
with a console.logged
wrapThatGift(30, 100, 2)
I got my answer!
108 inches by 66 inches
The original repo is here: https://github.com/Cerchie/crinkle-crinkle
We'd love to hear suggestions for improvement or other ways to solve! (If I go back to this, I'm going to create a toleranceFinder func to reduce that logic) Leave me comments to let me know what you think. :)
Top comments (8)
Could you not just use
Array.sort
for ordering the dimensions?Then the relevant piece of code would be:
Oh, by the way (not being trying to be snarky), you have the wrong variable in the tolerance calculation code. You're using
middle
length
instead ofwidth
longest
.And for that tolerance logic, you can
if/else
blocks instead of multiple separateif
blocks.Or just use a ternary expression:
And since this is just making sure the tolerance never goes above 2, you can just use this:
Also, you never actually use that calculated area.
As a nice-to-have, you can use
const
instead oflet
for most of the variables.If you want, you can return the length and width as an array or object of numbers instead of a string directly, so you could use it for other purposes as well, and just do the string templating when you log it.
So the code can go from this:
To this:
Hope this is useful!
P.S.
Add a language tag to you code blocks to get syntax highlighting.
Thanks Darryl! I didn't google anything, so I ended up with longer syntax for some of these -- your solution pares it down nicely! And the code block suggestion is great for blogging on DEV, thanks!
I believe 'middle' is the correct variable though-- and it's actually length that's wrong -- according to the equation we need the middle length for calculating the width tolerance and the longest length for the length tolerance -- so it should be
if (longest <= 4) { ...etc.
Ah, yes, I understand now.
Will update my comment. š
Wow
This looks like fun. I really want to try it myself. Never would have thought of doing this, but it's a great problem. Thanks for sharing!!!
Loved this. Great team work!!!
Suggestion: set overlap to Ā½ of length & width no matter what, then if it's greater than 2 set it back down to 2.
This is very interesting. I'm definitely signing up for that newsletter.
Nice work