Pen Settings

HTML

CSS

CSS Base

Vendor Prefixing

Add External Stylesheets/Pens

Any URL's added here will be added as <link>s in order, and before the CSS in the editor. If you link to another Pen, it will include the CSS from that Pen. If the preprocessor matches, it will attempt to combine them before processing.

+ add another resource

JavaScript

Babel includes JSX processing.

Add External Scripts/Pens

Any URL's added here will be added as <script>s in order, and run before the JavaScript in the editor. You can use the URL of any other Pen and it will include the JavaScript from that Pen.

+ add another resource

Packages

Add Packages

Search for and use JavaScript packages from npm here. By selecting a package, an import statement will be added to the top of the JavaScript editor for this package.

Behavior

Save Automatically?

If active, Pens will autosave every 30 seconds after being saved once.

Auto-Updating Preview

If enabled, the preview panel updates automatically as you code. If disabled, use the "Run" button to update.

Format on Save

If enabled, your code will be formatted when you actively save your Pen. Note: your code becomes un-folded during formatting.

Editor Settings

Code Indentation

Want to change your Syntax Highlighting theme, Fonts and more?

Visit your global Editor Settings.

HTML

              
                <div id="main">

	<svg width="200" height="200" viewBox="0 0 200 200" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" id="photo-frame">
		<g>
			<rect width="200" height="200" fill="white" />
			<g id="photo" clip-path="url(#clip0_8_89)">
				<rect width="180" height="180" transform="translate(10 10)" fill="#7DBDFC" />
				<g id="photo-car">
					<path id="photo-car-body" fill-rule="evenodd" clip-rule="evenodd" d="M25.6453 110.038C29.3626 110.038 32.3761 113.052 32.3761 116.769V116.769C32.3761 131.638 44.4299 143.692 59.2992 143.692C74.1684 143.692 86.2222 131.638 86.2222 116.769V116.769C86.2222 113.052 89.2357 110.038 92.953 110.038H106.415C110.132 110.038 113.145 113.052 113.145 116.769V116.769C113.145 131.638 125.199 143.692 140.068 143.692C154.938 143.692 166.991 131.638 166.991 116.769V116.769C166.991 113.052 170.005 110.038 173.722 110.038H187.184C198.336 110.038 207.376 119.079 207.376 130.231V190.808C207.376 201.96 198.336 211 187.184 211H-65.6838C-79.9692 211 -89.7373 196.572 -84.4319 183.308L-80.51 173.504C-65.1777 135.173 -28.0532 110.038 13.2304 110.038H25.6453Z" fill="#DB5D5D" />
					<path id="photo-go-faster-stripes" fill-rule="evenodd" clip-rule="evenodd" d="M190 162H10V166H190V162ZM190 168H10V172H190V168Z" fill="#FFF175" />
				</g>
				<path id="photo-front-body" d="M39 96.5C39 85.1782 48.1782 76 59.5 76V76C70.8218 76 80 85.1782 80 96.5V116.5C80 127.822 70.8218 137 59.5 137V137C48.1782 137 39 127.822 39 116.5V96.5Z" fill="#333333" />
				<path id="photo-back-body" d="M119.292 96.5C119.292 85.1782 128.47 76 139.792 76V76C151.114 76 160.292 85.1782 160.292 96.5V116.5C160.292 127.822 151.114 137 139.792 137V137C128.47 137 119.292 127.822 119.292 116.5V96.5Z" fill="#605DE4" />
				<path id="photo-front-left-arm" d="M43.6385 72.6096C49.7522 82.1687 47.7606 94.7884 39 102V102C39 102 39.5 90.5 30.5 75C25.852 66.9952 21.6041 60.9248 18.5827 56.9258C16.282 53.8807 16.9353 49.3757 20.1505 47.3194V47.3194C23.0168 45.4861 26.8266 46.3236 28.6598 49.19L43.6385 72.6096Z" fill="#333333" />
				<path id="photo-back-left-arm" d="M123.93 72.6096C130.044 82.1687 128.052 94.7884 119.292 102V102C119.292 102 119.792 90.5 110.792 75C106.144 66.9952 101.896 60.9248 98.8746 56.9258C96.5739 53.8807 97.2272 49.3757 100.442 47.3194V47.3194C103.309 45.4861 107.118 46.3236 108.952 49.19L123.93 72.6096Z" fill="#605DE4" />
				<rect id="photo-image 2" width="69" height="69" transform="matrix(-1 0 0 1 92 29)" fill="url(#pattern0)" />
				<rect id="photo-image 3" width="69" height="69" transform="matrix(-1 0 0 1 177 29)" fill="url(#pattern1)" />
				<path id="photo-front-right-arm" d="M80 103C80 112.5 58 86.4263 58 86.4263L95.4646 55.2751C97.6399 53.4663 100.87 53.7635 102.678 55.9388V55.9388C105.121 58.8762 103.665 63.6651 100.314 65.4994C96.6427 67.5091 92.2447 70.6709 88.5 75.5C80.3083 86.064 80 93.5 80 103Z" fill="#333333" />
				<path id="photo-back-right-arm" d="M160.292 103C160.292 112.5 138.292 86.4263 138.292 86.4263L175.756 55.2751C177.932 53.4663 181.161 53.7635 182.97 55.9388V55.9388C185.413 58.8762 183.957 63.6651 180.606 65.4994C176.935 67.5091 172.537 70.6709 168.792 75.5C160.6 86.064 160.292 93.5 160.292 103Z" fill="#605DE4" />
			</g>
		</g>
		<defs>
			<pattern id="pattern0" patternContentUnits="objectBoundingBox" width="1" height="1">
				<use xlink:href="#image0_8_89" transform="scale(0.005)" />
			</pattern>
			<pattern id="pattern1" patternContentUnits="objectBoundingBox" width="1" height="1">
				<use xlink:href="#image1_8_89" transform="scale(0.005)" />
			</pattern>
			<clipPath id="clip0_8_89">
				<rect width="180" height="180" fill="white" transform="translate(10 10)" />
			</clipPath>
			<image id="image0_8_89" data-name="image.png" width="200" height="200" xlink:href="" />
			<image id="image1_8_89" data-name="image.png" width="200" height="200" xlink:href="" />
		</defs>
	</svg>

	<svg width="700" height="437" viewBox="0 0 700 437" fill="none" xmlns="http://www.w3.org/2000/svg" id="rollercoaster">
		<g>
			<path id="camera" fill-rule="evenodd" clip-rule="evenodd" d="M378 32H382V49H385C385.552 49 386 49.4477 386 50V56C386 56.5523 385.552 57 385 57H375C374.448 57 374 56.5523 374 56V50C374 49.4477 374.448 49 375 49H378V32ZM380 55C381.105 55 382 54.1046 382 53C382 51.8954 381.105 51 380 51C378.895 51 378 51.8954 378 53C378 54.1046 378.895 55 380 55Z" fill="white" />

			<g id="supports">
				<rect id="supports_2" opacity="0.5" x="109" y="398" width="4" height="39" fill="white" />
				<rect id="supports_3" x="170" y="185" width="10" height="252" fill="white" />
				<rect id="supports_4" opacity="0.5" x="229" y="74" width="2" height="363" fill="white" />
				<rect id="supports_5" opacity="0.5" x="289" y="31" width="4" height="406" fill="white" />
				<rect id="supports_6" opacity="0.5" x="405" y="35" width="4" height="402" fill="white" />
				<rect id="supports_7" opacity="0.5" x="349" y="21" width="2" height="416" fill="white" />
				<rect id="supports_8" opacity="0.5" x="465" y="52" width="2" height="385" fill="white" />
				<rect id="supports_9" x="520" y="185" width="10" height="252" fill="white" />
				<rect id="supports_10" opacity="0.5" x="587" y="398" width="4" height="39" fill="white" />
			</g>
			<path id="exit-under" d="M106 363C106 355.268 99.732 349 92 349V349C84.268 349 78 355.268 78 363V437H106V363Z" fill="#333333" />
			<path id="right-rail" d="M349.707 6.96538C445.529 6.96538 523.208 85.9458 523.208 183.373C523.208 352.347 281.885 401.965 97 401.965" stroke="white" stroke-width="13.6231" />
			<path id="cover" fill-rule="evenodd" clip-rule="evenodd" d="M349 13.9725C348.911 13.9724 348.822 13.9723 348.733 13.9723V13.9731C255.205 14.2649 179.746 91.2123 179.746 185.751C179.746 266.825 236.37 318.752 310.589 350.274C249.006 365.277 180.397 371.187 118.362 371.187V393.897C191.753 393.897 275.437 385.811 347.11 363.797C419.602 387.13 504.932 395.892 579.638 395.892V373.182C516.443 373.182 446.397 366.763 383.915 350.766C459.823 319.893 518.254 268.112 518.254 185.751C518.254 91.2123 442.795 14.2649 349.267 13.9731V13.9723C349.178 13.9723 349.089 13.9724 349 13.9725ZM349 36.6825C268.192 36.8288 202.456 103.361 202.456 185.751C202.456 255.684 252.057 302.434 326.391 332.228C333.181 334.95 340.15 337.516 347.273 339.933C355.62 337.228 363.766 334.322 371.669 331.207C445.996 301.903 495.544 255.643 495.544 185.751C495.544 103.361 429.808 36.8288 349 36.6825Z" fill="#7DBDFC" />
			<path id="path" d="M706.5 380.5H585C432.5 380.5 194.5 342 194.5 185C194.5 94.5 268.5 26 350 26C431.5 26 505.5 94 505.5 185C505.5 318.173 326.5 382.5 114.5 382.5C65 382.5 32.8981 382.5 -9.5 382.5" stroke="black" />
			<path id="entrace-under" d="M595 363C595 355.268 601.268 349 609 349V349C616.732 349 623 355.268 623 363V437H595V363Z" fill="#333333" />
			<g id="car">
				<path id="car-body" fill-rule="evenodd" clip-rule="evenodd" d="M641 380C641.552 380 642 380.448 642 381V381C642 383.209 643.791 385 646 385C648.209 385 650 383.209 650 381V381C650 380.448 650.448 380 651 380H653C653.552 380 654 380.448 654 381V381C654 383.209 655.791 385 658 385C660.209 385 662 383.209 662 381V381C662 380.448 662.448 380 663 380H665C666.657 380 668 381.343 668 383V392C668 393.657 666.657 395 665 395H627.431C625.309 395 623.857 392.856 624.646 390.886L625.228 389.429C627.506 383.734 633.022 380 639.156 380H641Z" fill="#DB5D5D" />
				<rect id="front-body" x="643" y="375" width="6" height="9" rx="3" fill="#525D67" />
				<rect id="front-head" x="644" y="369" width="6" height="6" rx="3" fill="#E8E8E8" />
				<rect id="front-arm" x="645" y="369" width="2" height="8" rx="1" fill="#525D67" />
				<rect id="back-head" x="656" y="369" width="6" height="6" rx="3" fill="#E8E8E8" />
				<rect id="back-body" x="655" y="375" width="6" height="9" rx="3" fill="#525D67" />
				<rect id="back-arm" x="657" y="369" width="2" height="8" rx="1" fill="#525D67" />
				<path id="Vector 1" d="M633 387.5C631 387.5 630 385 629 384" stroke="#FFF175" />
				<path id="Vector 2" d="M633 389.5C629 389.5 629 386.5 627.5 385.5" stroke="#FFF175" />
				<rect id="Rectangle 22" x="633" y="387" width="35" height="1" fill="#FFF175" />
				<rect id="Rectangle 23" x="633" y="389" width="35" height="1" fill="#FFF175" />
			</g>
			<rect id="entrance-cover" x="623" y="349" width="77" height="88" fill="#7DBDFC" />
			<rect id="exit-cover" y="349" width="78" height="88" fill="#7DBDFC" />
			<path id="exit-over" d="M92 349C84.268 349 78 355.268 78 363V437H92V349V349Z" fill="url(#paint0_linear_2_21)" />
			<path id="entrace-over" d="M609 349C616.732 349 623 355.268 623 363V437H609V349V349Z" fill="url(#paint1_linear_2_21)" />
			<path id="left-rail" d="M350.293 6.96538C254.471 6.96538 176.792 85.9458 176.792 183.373C176.792 352.347 418.115 401.965 603 401.965" stroke="white" stroke-width="13.6231" />
			<circle id="camera-flash" cx="380" cy="53" r="10" stroke="white" stroke-width="2" />
		</g>
		<defs>
			<linearGradient id="paint0_linear_2_21" x1="78" y1="393" x2="92" y2="393" gradientUnits="userSpaceOnUse">
				<stop offset="0.302083" stop-color="#333333" />
				<stop offset="1" stop-color="#333333" stop-opacity="0" />
			</linearGradient>
			<linearGradient id="paint1_linear_2_21" x1="623" y1="393" x2="609" y2="393" gradientUnits="userSpaceOnUse">
				<stop offset="0.302083" stop-color="#333333" />
				<stop offset="1" stop-color="#333333" stop-opacity="0" />
			</linearGradient>
		</defs>
	</svg>

</div>
              
            
!

CSS

              
                body{
	background-color:#7DBDFC;
}

#path{
	stroke:none;
}

#main{
	position:absolute;
	top:50%;
	left:50%;
	width:551px;
	height:437px;
	transform: translate(-50%, -50%);
	
	#rollercoaster{
		display:block;
		margin-left: 50%;
		transform: translate(-50%);
	}
	
	#photo-frame{
		position:absolute;
		top:15%;
		left:-22%;
		transform: rotate(5deg);
		box-shadow:0 0 20px #208BF3;
		
	}

}

#camera-flash{
	opacity:0;
}


              
            
!

JS

              
                /**
Submission for the GSAP weekly challenge LOOP!

A animation of a Roller Coaster car doing a loop using GSAP.

A souvenir photograph is taken at the loops peak which rider can purchase at the kiosk after the ride!
*/


gsap.registerPlugin(MotionPathPlugin);

const coasterTimeline = gsap.timeline({ repeat: -1 });

/**
 * Generates a string containing a combination of letters and numbers
 * @returns string
 */
const randomSeed = () => {
  return Math.random().toString(36).slice(2);
};

/**
 * Returns a random hex colour from a list of colours
 * @param frontRiderColour if provided, the returned colour will not be the same as the one provided
 * @returns string
 */
const randomColour = (frontRiderColour = false) => {
  const colours = [
    "#D56C0C",
    "#605DE4",
    "#238D80",
    "#333333",
    "#E9B729",
    "#9A3E91",
  ];
  let randomColour = colours[Math.floor(Math.random() * colours.length)];

  while (randomColour === frontRiderColour) {
    randomColour = colours[Math.floor(Math.random() * colours.length)];
  }

  return randomColour;
};

/**
 * Changes the faces abd colours of both riders
 */
const changeRiders = () => {
  // Get all riders
  const riders = document.querySelectorAll("#photo-frame image");

  // Change front rider head
  riders[0].setAttribute(
    "href",
    `https://avatars.dicebear.com/api/big-smile/${randomSeed()}.svg?w=200&scale=90`
  );

  // Get a new colour for the front rider
  const frontRiderColour = randomColour();
  // Change front riders clothing colour
  document
    .getElementById("photo-front-body")
    .setAttribute("fill", frontRiderColour);
  document
    .getElementById("photo-front-left-arm")
    .setAttribute("fill", frontRiderColour);
  document
    .getElementById("photo-front-right-arm")
    .setAttribute("fill", frontRiderColour);

  // Change back rider head
  riders[1].setAttribute(
    "href",
    `https://avatars.dicebear.com/api/big-smile/${randomSeed()}.svg?w=200&scale=90`
  );

  // Get a new colour for the back rider (not the same as the front rider)
  const backRiderColour = randomColour(frontRiderColour);
  // Change front riders clothing colour
  document
    .getElementById("photo-back-body")
    .setAttribute("fill", backRiderColour);
  document
    .getElementById("photo-back-left-arm")
    .setAttribute("fill", backRiderColour);
  document
    .getElementById("photo-back-right-arm")
    .setAttribute("fill", backRiderColour);
};

const coasterDuration = 5;

// -------------------------
// Start of coaster timeline
// -------------------------
coasterTimeline.set("#camera", { scale: 1 });
coasterTimeline.to("#car", {
  motionPath: {
    path: "#path",
    align: "#path",
    alignOrigin: [0.5, 0.4],
    autoRotate: 180,
    start: 0,
    end: 1,
  },
  transformOrigin: "50% 50%",
  duration: coasterDuration,
  ease: "slow(0.01, 0.9, false)",
});

// Timeline labels
coasterTimeline.addLabel("slowMoStart", "<35%");
coasterTimeline.addLabel("cameraFlash", "<40%");
coasterTimeline.addLabel("slowMoEnd", "<75%");

// Photo fade out
coasterTimeline.add(
  gsap.to("#photo", { duration: 1, opacity: 0, onComplete: () => changeRiders() }),
  "<0%"
);

// Camera flash
coasterTimeline.add(
  gsap.fromTo(
    "#camera-flash",
    { strokeWidth: 30, scale: 0, transformOrigin: "50% 50%", opacity: 1 },
    {
      duration: 1,
      strokeWidth: 2,
      scale: 4,
      transformOrigin: "50% 50%",
      opacity: 0,
      ease: "power3.out",
    }
  ),
  "cameraFlash"
);

// Camera bounce
coasterTimeline.add(
  gsap.fromTo(
    "#camera",
    { scale: 0.5 },
    { duration: 1, scale: 1, ease: "elastic" }
  ),
  "cameraFlash"
);

// Photo fade in
coasterTimeline.add(
  gsap.to("#photo", { duration: 0.3, opacity: 1, delay: 0.3 }),
  "cameraFlash"
);

// Riders arm animation(s)
coasterTimeline.add(
  gsap.from(["#front-arm", "#back-arm"], {
    duration: coasterDuration / 2,
    rotation: 90,
    transformOrigin: "50% 100%",
    ease: "elastic.out",
  }),
  "slowMoStart"
);

coasterTimeline.add(
  gsap.to(["#front-arm", "#back-arm"], {
    duration: coasterDuration / 5,
    rotation: 90,
    transformOrigin: "50% 100%",
    ease: "none",
  }),
  "slowMoEnd"
);

              
            
!
999px

Console