Procedural dynamic trees driven by local wind speed

The following code adds procedural, animated trees to the footer of my art section page. The trees are generated without the use of textures, or canvas, and make use of my weather data to react to wind speed and wind direction. This is reflected in their gentle, or not so gentle swaying. The trees of my little forest have three layers - the stem, primary branches, and secondary branches. The animation is procedural and driven by real time weather variables. It includes moving the thinner parts of the trees more than the thicker parts. Additionally the trees grow, fade, and re-grow every minute.

You may notice that some variable names refer to vines, this is a relic of how the project started. You can view the trees live in action on my art page, and see how I set the weather data variables to be retrieved in local storage in my blog post about weather effects.

HTML:

JAVASCRIPT:

<script>
document.addEventListener('DOMContentLoaded', function () { 
const subString = '/art';
const getURL = window.location.href;  
if (getURL.includes(subString)) { 
//mobile check
const headerActions = document.querySelector('.header-actions');
let isTouch = false;
function checkHeader() {
const styles = window.getComputedStyle(headerActions);
isTouch = styles.getPropertyValue('display') !== 'flex';
}
checkHeader();
if (isTouch === false) {
const footer = document.getElementById('footer-sections');
const bottomSeparator = footer.querySelector('.horizontalrule-block');
const vinesMainContainer = document.createElement('div');
vinesMainContainer.id = "vinesMainContainer";
vinesMainContainer.classList.add('vinesMainContainer');  
const vinesContainer = document.createElement('div');
vinesContainer.id = "vinesContainer";
vinesContainer.classList.add('vinesContainer'); 
vinesMainContainer.appendChild(vinesContainer);
bottomSeparator.appendChild(vinesMainContainer);
//initial setup
const drawSpeed = 35;
const vinesDuration = 60 * 1000;
const maxVineCount = 15;
const hardMaxVineCount = 25;
let vineCount = 0;
let minInterval = 30;
let maxInterval = 500;
const minStartThickness = 4;
const maxStartThickness = 6;
let randomInterval = 100;
let lastVineStartX = [];
const minDistanceBetweenVines = 50;
const maxAttempts = minDistanceBetweenVines / 2;
const minVineHeight = 0.2;
const maxVineHeight = 0.55;
const splitOccurStart = 0.1;
const splitOccurEnd = 0.8;
let windSpeed = 15; 
let windDirection = 360;
const storedWeatherData = localStorage.getItem('weatherData');
if (storedWeatherData) {
const weatherData = JSON.parse(storedWeatherData);
const weather = weatherData.weather;
windDirection = weather.winddirection;
windSpeed = weather.windspeed;
}
//swaying animation
function applyWindEffect(svg) {
const vineSegments = svg.querySelectorAll('polyline'); // Get all vine segments
const mainVineSegments = Array.from(vineSegments).filter(segment => segment.classList.contains('main-vine')); // Filter for main vine segments
const baseSegment = mainVineSegments[0]; // The base of the main vine
const basePoint = baseSegment.getAttribute('points').split(' ')[0].split(',').map(Number);
const maxRotation = windSpeed * 0.00008; // Max swaying angle
let swayAngle = 0;
const swayDirection = windDirection >= 180 ? -1 : 1;
const swayIncrement = 0.07; // Adjust this value as needed
function sway() {
swayAngle += 0.03;
const rotationAngle = Math.sin(swayAngle) * maxRotation;
for (let i = 0; i < mainVineSegments.length; i++) {
const segment = mainVineSegments[i];
const group = segment.closest('g');
const subgroups = group ? group.querySelectorAll('g') : [];
let points = segment.getAttribute('points').split(' ').map(point => point.split(',').map(Number));
points = points.map((point, index) => {
const [x, y] = point;
const dx = x - basePoint[0];
const dy = y - basePoint[1];
const currentRotation = rotationAngle * (1 + swayIncrement * i);
const newX = basePoint[0] + (dx * Math.cos(currentRotation * swayDirection) - dy * Math.sin(currentRotation * swayDirection));
const newY = basePoint[1] + (dx * Math.sin(currentRotation * swayDirection) + dy * Math.cos(currentRotation));
return [newX, newY];
});
if (i > 0) {
points[0] = mainVineSegments[i - 1].getAttribute('points').split(' ').map(point => point.split(',').map(Number)).slice(-1)[0];
} else {
points[0] = basePoint;
}
segment.setAttribute('points', points.map(point => point.join(',')).join(' ')); 
//move branches
subgroups.forEach(subgroup => {
const subgroupPolylines = subgroup.querySelectorAll('polyline');
subgroupPolylines.forEach(polyline => {
const subgroupPoints = polyline.getAttribute('points') ? polyline.getAttribute('points').split(' ').map(point => point.split(',').map(Number)) : [];
const isSecondarySplit = polyline.classList.contains('secondary-split');
const newSubgroupPoints = subgroupPoints.map((point) => {
const [x, y] = point;
const dx = x - basePoint[0];
const dy = y - basePoint[1];
// Use different logic for secondary splits
const currentRotation = isSecondarySplit ? rotationAngle * swayDirection : rotationAngle * (1 + swayIncrement * i); 
const newX = basePoint[0] + (dx * Math.cos(currentRotation) - dy * Math.sin(currentRotation));
const newY = basePoint[1] + (dx * Math.sin(currentRotation) + dy * Math.cos(currentRotation));
return [newX, newY];
});
if (newSubgroupPoints.length > 0) {
polyline.setAttribute('points', newSubgroupPoints.map(point => point.join(',')).join(' '));
}
});
});
//end branches move  
}
requestAnimationFrame(sway);
}
sway();
}
//create the trees
function createVinesSVG(id) {
const svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg');
svg.setAttribute('id', id);
svg.setAttribute('class', 'vine');
vinesContainer.appendChild(svg);
return svg;
}
function createVines() {
const uniqueId = `vine-${Date.now()}`;
const svg = createVinesSVG(uniqueId);
let startX;
let attempts = 0;
// Try to find a valid position for the vine's startX
do {
startX = Math.random() * svg.clientWidth;
attempts++;
} while (attempts < maxAttempts && lastVineStartX.some(prevX => Math.abs(prevX - startX) < minDistanceBetweenVines));
let startY = svg.clientHeight; 
let maxHeight = Math.random() * (svg.clientHeight * maxVineHeight) + (svg.clientHeight * minVineHeight);
let segments = Math.floor(maxHeight / (svg.clientHeight / 40));
const startThickness = Math.random() * (maxStartThickness - minStartThickness) + minStartThickness;
let x = startX;
let y = startY;
lastVineStartX.push(startX);
if (lastVineStartX.length > maxVineCount - 1) {
lastVineStartX.shift();
}
// Draw segments over time
let currentSegment = 0;
let splitCount = 0;
let drawInterval = setInterval(() => {
if (currentSegment < segments) {
let polyline = document.createElementNS('http://www.w3.org/2000/svg', 'polyline');
polyline.setAttribute('fill', 'none');
let thickness = Math.max(startThickness - (currentSegment * (maxStartThickness / segments)), maxStartThickness / 3.3);
polyline.setAttribute('stroke-width', `${thickness}px`);
polyline.setAttribute('class', 'main-vine'); 
let mainGroup = document.createElementNS('http://www.w3.org/2000/svg', 'g');
mainGroup.appendChild(polyline);
svg.appendChild(mainGroup);
let points = `${x},${y}`;
y -= (maxHeight / segments);
x += (Math.random() - 0.5) * 3; // Random horizontal displacement
points += ` ${x},${y}`;
polyline.setAttribute('points', points);
// Determine if a split should occur
let splitStartThreshold = Math.floor(segments * splitOccurStart);
let splitEndThreshold = Math.floor(segments * splitOccurEnd);
let splitDirection = Math.random() < 0.5 ? 'left' : 'right';
if (currentSegment > splitStartThreshold && currentSegment < splitEndThreshold && currentSegment % 2 === 0  && Math.random() > 0.45) {
drawSplit({ x, y, direction: splitDirection }, svg, splitCount, thickness, mainGroup);
splitCount++;
}
currentSegment++;
} else {
clearInterval(drawInterval);
setTimeout(() => {
applyWindEffect(svg);
}, 3000);
setTimeout(() => {
vinesContainer.removeChild(svg);
vineCount--;
}, vinesDuration);
}
}, drawSpeed);
}
// Splitting for primary branches
function drawSplit(split, svg, splitCount, stemThickness, mainGroup) {
let x = split.x;
let y = split.y;
let splitSegments = 4 + Math.floor(Math.random() * 3);
let maxSplitLength = (svg.clientHeight / 30) * (4 - splitCount);
let minSplitLength = (svg.clientHeight / 30) * 2;
let splitHeight = Math.random() * (maxSplitLength - minSplitLength) + minSplitLength;
let maxHorizontalDisplacement = Math.tan(Math.PI / 4.5) * splitHeight;
let currentSplitSegment = 0;
let thickness = stemThickness / 1.3; //starting branch thickness
let splitStartThreshold = Math.floor(splitSegments * 0.2);
let splitEndThreshold = Math.floor(splitSegments * 0.5);
let splitGroup = document.createElementNS('http://www.w3.org/2000/svg', 'g');
mainGroup.appendChild(splitGroup);
let splitDrawInterval = setInterval(() => {
if (currentSplitSegment < splitSegments) {
let points = `${x},${y}`;
let horizontalDisplacement = (Math.random() - 0.5) * maxHorizontalDisplacement;
if (split.direction === 'left') {
x -= Math.abs(horizontalDisplacement);
} else {
x += Math.abs(horizontalDisplacement);
}
y -= splitHeight / splitSegments;
points += ` ${x},${y}`;
let splitPolyline = document.createElementNS('http://www.w3.org/2000/svg', 'polyline');
splitPolyline.setAttribute('fill', 'none');
splitPolyline.setAttribute('points', points);
splitPolyline.setAttribute('stroke-width', `${thickness}px`); 
splitPolyline.setAttribute('class', 'primary-split'); 
splitGroup.appendChild(splitPolyline);
//svg.appendChild(splitPolyline);
thickness = Math.max(thickness - ((stemThickness - 1) / splitSegments), 1);
// Randomly trigger secondary split
if (currentSplitSegment >= splitStartThreshold && currentSplitSegment <= splitEndThreshold && Math.random() > 0.43) {
drawSecondarySplit({ x, y, direction: Math.random() < 0.5 ? 'left' : 'right' }, svg, splitCount + 1, thickness, splitGroup);
}
currentSplitSegment++;
} else {
clearInterval(splitDrawInterval);
}
}, drawSpeed * 2);
}
// secondary branches
function drawSecondarySplit(split, svg, splitCount, parentThickness, splitGroup) {
let x = split.x;
let y = split.y;
let splitSegments = 2;
let maxSplitLength = (svg.clientHeight / 50) * (3 - splitCount);
let minSplitLength = (svg.clientHeight / 60) * 1;
let splitHeight = Math.random() * (maxSplitLength - minSplitLength) + minSplitLength;
let maxHorizontalDisplacement = Math.tan(Math.PI / 8) * splitHeight;
let currentSplitSegment = 0;
let thickness = parentThickness / 1.7; 
const secondaryGroup = document.createElementNS('http://www.w3.org/2000/svg', 'g');
splitGroup.appendChild(secondaryGroup);
let splitDrawInterval = setInterval(() => {
if (currentSplitSegment < splitSegments) {
let points = `${x},${y}`;
let horizontalDisplacement = (Math.random() - 0.5) * maxHorizontalDisplacement;
if (split.direction === 'left') {
x -= Math.abs(horizontalDisplacement);
} else {
x += Math.abs(horizontalDisplacement);
}
y -= splitHeight / splitSegments;
points += ` ${x},${y}`;
let secondaryPolyline = document.createElementNS('http://www.w3.org/2000/svg', 'polyline');
secondaryPolyline.setAttribute('fill', 'none');
secondaryPolyline.setAttribute('points', points);
secondaryPolyline.setAttribute('stroke-width', `${thickness}px`);
secondaryPolyline.setAttribute('class', 'secondary-split');
secondaryGroup.appendChild(secondaryPolyline);
currentSplitSegment++;
} else {
clearInterval(splitDrawInterval);
}
}, drawSpeed * 4);
}
//animate
function triggerRandomVines() {
if (vineCount < maxVineCount) {
vineCount++;
createVines();
}
randomInterval = Math.random() * (maxInterval - minInterval) + minInterval;
setTimeout(triggerRandomVines, randomInterval);
}
setTimeout(triggerRandomVines, 2000);
//exit
}
}
});
</script>
CSS:

@keyframes vineFade {
0% {opacity: 1; stroke: #000000; filter: blur(0.0px)}
95% {opacity: 1; stroke: #000000; filter: blur(0.0px)}
100% {opacity: 0; filter: blur(6px); stroke: #464646;}
}

//VINES TREES FOREST//
#vinesMainContainer {
  position: absolute;
  display: flex;
  justify-content: center;
  align-items: center;
  pointer-events: none;
  width: 75%;
  height: 200px;
  transform: translateX(-50%);
  align-items: center;
  bottom: 50%;
  left: 50%;
  padding: 0; 
}
#vinesContainer {
  position: absolute;
  pointer-events: none;
  width: 100%;
  height: 100%;
  top: 0;
  -webkit-mask-image: linear-gradient(90deg, rgba(0,0,0,0) 0%, rgba(0,0,0,1) 10%, rgba(0,0,0,1) 90%, rgba(0,0,0,0) 100%);
  mask-image: linear-gradient(90deg, rgba(0,0,0,0) 0%, rgba(0,0,0,1) 10%, rgba(0,0,0,1) 90%, rgba(0,0,0,0) 100%);
}
.vine {
  position: absolute;
  pointer-events: none;
  //filter: blur(0.0px);
  stroke: #000000;
  width: 100%;
  height: 100%;
  top: 0%;
  stroke-linecap: round;
  stroke-linejoin: round;
  animation: vineFade 59s forwards;
  -webkit-mask-image: linear-gradient(0deg, rgba(0,0,0,1) 0%, rgba(0,0,0,1) 7%, rgba(0,0,0,1) 40%, rgba(0,0,0,0) 60%);
  mask-image: linear-gradient(0deg, rgba(0,0,0,1) 0%, rgba(0,0,0,1) 7%, rgba(0,0,0,1) 40%, rgba(0,0,0,0) 60%);
}
Previous
Previous

Moving car indicator with jump

Next
Next

Procedural dynamic Lightning effect