Compare commits
1 Commits
assignment
...
assignment
| Author | SHA1 | Date | |
|---|---|---|---|
| 3f3daab726 |
23
README.md
23
README.md
@@ -1,8 +1,17 @@
|
|||||||
# Advent of Code 2022 - Assignment8 javascript
|
# Advent of Code 2022 - assignment 1
|
||||||
## Description
|
This is the first assignment for Advent of Code 2022. The provided solution uses 3 variables: currentScore, highestCalorieScore, currentElf, highestElf.
|
||||||
This problem is all about walking through a grid. In this solution, we look through an entire column and row until we find a node with height <= height.
|
It is complexity O(n+m) since it loops through every line exactly once (n) and the iterates for recalculating highscore (m).
|
||||||
|
The GUI also provides a checkbox to toggle top3 or top1.
|
||||||
|
|
||||||
## References
|
## Solution description
|
||||||
- https://www.w3schools.com/jsref/jsref_charat.asp
|
Every iteration the score is added to the currentScore. If an empty line is detected we check if currentscore exceeds one of our highestCalorieScore, if yes we assign currentElf to highestElf.
|
||||||
- https://stackabuse.com/javascript-check-if-variable-is-a-number/
|
|
||||||
- https://www.w3schools.com/jsref/jsref_abs.asp
|
## references
|
||||||
|
- https://adventofcode.com/2022/day/1
|
||||||
|
- https://stackoverflow.com/questions/17595361/how-to-read-text-line-by-line-in-a-html-text-area
|
||||||
|
- https://bobbyhadz.com/blog/javascript-get-value-of-textarea
|
||||||
|
- https://stackoverflow.com/questions/154059/how-do-i-check-for-an-empty-undefined-null-string-in-javascript
|
||||||
|
- https://www.geeksforgeeks.org/convert-a-string-to-an-integer-in-javascript/#:~:text=In%20JavaScript%20parseInt()%20function,argument%20of%20parseInt()%20function.
|
||||||
|
- https://stackoverflow.com/questions/1230233/how-to-find-the-sum-of-an-array-of-numbers
|
||||||
|
- https://www.w3schools.com/js/js_break.asp
|
||||||
|
- https://2ality.com/2018/12/creating-arrays.html
|
||||||
@@ -8,8 +8,8 @@
|
|||||||
<body>
|
<body>
|
||||||
<p>Assignment:</p>
|
<p>Assignment:</p>
|
||||||
<textarea rows="15" cols="50" id="assignment"></textarea>
|
<textarea rows="15" cols="50" id="assignment"></textarea>
|
||||||
<input type="checkbox" id="algorithm" />
|
<input type="checkbox" id="algorithm"/>
|
||||||
<label for="algorithm">Highest scenic score?</label>
|
<label for="algorithm">TOP 3</label>
|
||||||
|
|
||||||
<p>Answer:</p>
|
<p>Answer:</p>
|
||||||
<div id="answer">Provide input first</div>
|
<div id="answer">Provide input first</div>
|
||||||
|
|||||||
138
script.js
138
script.js
@@ -2,13 +2,15 @@ const ASSIGNMENT = 'assignment';
|
|||||||
const ANSWER = 'answer';
|
const ANSWER = 'answer';
|
||||||
const ALGORITHM_CHECKBOX = 'algorithm';
|
const ALGORITHM_CHECKBOX = 'algorithm';
|
||||||
const NEWLINE_CHARACTER = '\n';
|
const NEWLINE_CHARACTER = '\n';
|
||||||
|
const TOP_1 = 1;
|
||||||
|
const TOP_N = 3;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Main function
|
* Main function
|
||||||
*/
|
*/
|
||||||
window.onload = function () {
|
window.onload = function() {
|
||||||
document.getElementById(ASSIGNMENT).addEventListener("input", calculateAnswer);
|
document.getElementById(ASSIGNMENT).addEventListener("input", calculateAnswer);
|
||||||
document.getElementById(ALGORITHM_CHECKBOX).addEventListener("click", calculateAnswer);
|
document.getElementById(ALGORITHM_CHECKBOX).addEventListener("click", calculateAnswer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -18,117 +20,51 @@ window.onload = function () {
|
|||||||
function calculateAnswer(event) {
|
function calculateAnswer(event) {
|
||||||
console.info("Calculating answer for input...");
|
console.info("Calculating answer for input...");
|
||||||
let assignment = document.getElementById(ASSIGNMENT).value;
|
let assignment = document.getElementById(ASSIGNMENT).value;
|
||||||
let getHighestScenicScore = document.getElementById(ALGORITHM_CHECKBOX).checked;
|
let top3 = document.getElementById(ALGORITHM_CHECKBOX).checked;
|
||||||
let answer = algorithm(assignment, getHighestScenicScore);
|
let answer = algorithm(assignment, (top3)? TOP_N: 1);
|
||||||
|
|
||||||
document.getElementById(ANSWER).innerText = answer;
|
document.getElementById(ANSWER).innerText = answer;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Calculate the answer to assignment.
|
* Calculate the answer to assignment1 with 2 variables, count and highest.
|
||||||
* @param assignment the input from the assignment.
|
* @param assignment the input from the assignment.
|
||||||
* @return string the answer
|
* @param topn the
|
||||||
|
* @return string the elf with the highest calory count
|
||||||
*/
|
*/
|
||||||
function algorithm(assignment, getHighestScenicScore) {
|
function algorithm(assignment, topn) {
|
||||||
let lines = assignment.trim().split(NEWLINE_CHARACTER);
|
let currentCalorieScore = 0;
|
||||||
console.info("Linecount:" + lines.length);
|
let highestCalorieScores = Array.from({length: topn}, () => (0));
|
||||||
|
let currentElf = 1;
|
||||||
|
|
||||||
let visibleTreeCount = 0;
|
let lines = assignment.trim().split(NEWLINE_CHARACTER);
|
||||||
let highestScenicScore = 0;
|
console.info("Linecount:" + lines.length);
|
||||||
|
|
||||||
for (let i = 0; i < lines.length; i++) {
|
for(var i = 0; i < lines.length; i++){
|
||||||
let row = lines[i].trim();
|
let line = lines[i];
|
||||||
|
|
||||||
for (let j = 0; j < row.length; j++) {
|
if(!line && currentCalorieScore > 0) { // Check for empty line and skip subsequent emptylines
|
||||||
if (i <= 0 || i >= lines.length - 1 || j <= 0 || j >= row.length - 1) { // Edge node
|
console.debug("Calculated score for elf " + currentElf + ": " + currentCalorieScore);
|
||||||
//console.debug("Edge node found.");
|
|
||||||
visibleTreeCount++;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
let height = parseInt(lines[i].charAt(j));
|
for(let j=0; j < highestCalorieScores.length; j++) {
|
||||||
if (!walkRowVisible(height, j, row) && !walkColumnVisible(height, i, j, lines)) {
|
if(currentCalorieScore > highestCalorieScores[j]) {
|
||||||
console.log("Found invisible tree, height:" + height + ", i:" + i + ", j:" + j);
|
console.debug("New highest #"+(j+1) + " elf found, with score:" + currentCalorieScore);
|
||||||
visibleTreeCount--;
|
let oldHighestCalorieScore = highestCalorieScores[j];
|
||||||
}
|
highestCalorieScores[j] = currentCalorieScore;
|
||||||
|
currentCalorieScore = oldHighestCalorieScore;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (getHighestScenicScore) {
|
// Reset values
|
||||||
let scenicScore = walkRowCount(height, j, row) * walkColumnCount(height, i, j, lines);
|
currentCalorieScore = 0;
|
||||||
if (scenicScore > highestScenicScore) {
|
currentElf++;
|
||||||
console.debug("Found new highest scenic score: " + scenicScore);
|
continue;
|
||||||
highestScenicScore = scenicScore;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
let calorieScore = parseInt(line);
|
||||||
// Tree is on edge.
|
currentCalorieScore += calorieScore;
|
||||||
visibleTreeCount++;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (getHighestScenicScore) {
|
console.debug(highestCalorieScores);
|
||||||
return "Highest scenic score: " + highestScenicScore;
|
return "The highest of sum of calorie amount: " + highestCalorieScores.reduce(function(a, b) { return a + b; }, 0);
|
||||||
}
|
|
||||||
|
|
||||||
return "Amount of visible trees in the grid: " + visibleTreeCount;
|
|
||||||
}
|
|
||||||
|
|
||||||
function walkColumnCount(height, index, rowIndex, column) {
|
|
||||||
return walkCount(height, index, column, false, rowIndex);
|
|
||||||
}
|
|
||||||
|
|
||||||
function walkRowCount(height, index, row) {
|
|
||||||
return walkCount(height, index, row, true, -1);
|
|
||||||
}
|
|
||||||
|
|
||||||
function walkColumnVisible(height, index, rowIndex, column) {
|
|
||||||
return walkVisible(height, index, column, false, rowIndex);
|
|
||||||
}
|
|
||||||
|
|
||||||
function walkRowVisible(height, index, row) {
|
|
||||||
return walkVisible(height, index, row, true, -1);
|
|
||||||
}
|
|
||||||
|
|
||||||
function walkCount(height, index, line, isRow, rowIndex) {
|
|
||||||
let visibleLeftCount = __walkHelper(height, index, line, isRow, -1, rowIndex, 0);
|
|
||||||
let visibleRightCount = __walkHelper(height, index, line, isRow, 1, rowIndex, 0);
|
|
||||||
|
|
||||||
return visibleLeftCount * visibleRightCount;
|
|
||||||
}
|
|
||||||
|
|
||||||
function walkVisible(height, index, line, isRow, rowIndex) {
|
|
||||||
let visibleLeft = __walkHelper(height, index, line, isRow, -1, rowIndex);
|
|
||||||
let visibleRight = __walkHelper(height, index, line, isRow, 1, rowIndex);
|
|
||||||
|
|
||||||
return visibleLeft || visibleRight;
|
|
||||||
}
|
|
||||||
|
|
||||||
function __walkHelper(height, index, line, isRow, direction, rowIndex, count) {
|
|
||||||
if (direction == 0) {
|
|
||||||
console.error("Invalid direction 0 in __walkhelper");
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
direction = direction / Math.abs(direction);
|
|
||||||
|
|
||||||
// Edge reached
|
|
||||||
if ((direction < 0 && index <= 0) || (direction > 0 && index >= line.length - 1)) {
|
|
||||||
return (count == undefined) ? true : count;
|
|
||||||
}
|
|
||||||
|
|
||||||
let nextIndex = index + direction;
|
|
||||||
let nextHeightCharacter = (isRow) ? line.charAt(nextIndex) : line[nextIndex].trim().charAt(rowIndex);
|
|
||||||
let nextHeight = parseInt(nextHeightCharacter);
|
|
||||||
|
|
||||||
if (isNaN(height) || isNaN(nextHeightCharacter)) {
|
|
||||||
console.error("Invalid height found, height:" + height + ", nextHeight: " + nextHeightCharacter);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (height <= nextHeight) {
|
|
||||||
if (count == undefined)
|
|
||||||
return false;
|
|
||||||
return count + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return __walkHelper(height, nextIndex, line, isRow, direction, rowIndex, (count !=undefined)? count + 1 : undefined);
|
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user