Animated Curved Nav Using JavaScript

Animated Curved Nav Using JavaScript
Code Snippet:Curved Nav
Author: Taha Shashtari
Published: January 28, 2024
Last Updated: February 3, 2024
Downloads: 527
License: MIT
Edit Code online: CodeHim
Read More

This code creates a visually engaging curved navigation. The JavaScript-based animation smoothly transitions between navigation items. Helpful for adding an eye-catching interactive element to your webpage.

The core functionality involves smoothly transitioning between navigation items as the user interacts with the curved menu. This code helps create engaging and interactive user interfaces, adding a touch of creativity to website navigation.

How to Create Animated Curved Nav Using Javascript

1. First of all, include the Veloxi library by adding the following CDN link to your HTML’s <head> section:

  1. <script src='https://unpkg.com/veloxi/dist/veloxi.min.js'></script>

2. Next, structure your HTML with a container for the curved navigation and placeholders for content and navigation items:

  1. <div class="curved-nav">
  2. <div class="content-wrapper">
  3. <div
  4. class="content-item"
  5. data-vel-plugin="CurvedNavPlugin"
  6. data-vel-view="contentItem"
  7. >
  8. A
  9. </div>
  10. <div
  11. class="content-item"
  12. data-vel-plugin="CurvedNavPlugin"
  13. data-vel-view="contentItem"
  14. >
  15. B
  16. </div>
  17. <div
  18. class="content-item"
  19. data-vel-plugin="CurvedNavPlugin"
  20. data-vel-view="contentItem"
  21. >
  22. C
  23. </div>
  24. <div
  25. class="content-item"
  26. data-vel-plugin="CurvedNavPlugin"
  27. data-vel-view="contentItem"
  28. >
  29. D
  30. </div>
  31. <div
  32. class="content-item"
  33. data-vel-plugin="CurvedNavPlugin"
  34. data-vel-view="contentItem"
  35. >
  36. E
  37. </div>
  38. <div
  39. class="content-item"
  40. data-vel-plugin="CurvedNavPlugin"
  41. data-vel-view="contentItem"
  42. >
  43. F
  44. </div>
  45. <div
  46. class="content-item"
  47. data-vel-plugin="CurvedNavPlugin"
  48. data-vel-view="contentItem"
  49. >
  50. G
  51. </div>
  52. <div
  53. class="content-item"
  54. data-vel-plugin="CurvedNavPlugin"
  55. data-vel-view="contentItem"
  56. >
  57. H
  58. </div>
  59. <div
  60. class="content-item"
  61. data-vel-plugin="CurvedNavPlugin"
  62. data-vel-view="contentItem"
  63. >
  64. I
  65. </div>
  66. <div
  67. class="content-item"
  68. data-vel-plugin="CurvedNavPlugin"
  69. data-vel-view="contentItem"
  70. >
  71. J
  72. </div>
  73. </div>
  74. <div class="nav-container-wrapper">
  75. <div class="nav-container">
  76. <div
  77. class="nav-items"
  78. data-vel-plugin="CurvedNavPlugin"
  79. data-vel-view="itemsContainer"
  80. data-vel-data-active-index="0"
  81. >
  82. <div
  83. class="nav-item"
  84. data-vel-plugin="CurvedNavPlugin"
  85. data-vel-view="item"
  86. >
  87. A
  88. </div>
  89. <div
  90. class="nav-item"
  91. data-vel-plugin="CurvedNavPlugin"
  92. data-vel-view="item"
  93. >
  94. B
  95. </div>
  96. <div
  97. class="nav-item"
  98. data-vel-plugin="CurvedNavPlugin"
  99. data-vel-view="item"
  100. >
  101. C
  102. </div>
  103. <div
  104. class="nav-item"
  105. data-vel-plugin="CurvedNavPlugin"
  106. data-vel-view="item"
  107. >
  108. D
  109. </div>
  110. <div
  111. class="nav-item"
  112. data-vel-plugin="CurvedNavPlugin"
  113. data-vel-view="item"
  114. >
  115. E
  116. </div>
  117. <div
  118. class="nav-item"
  119. data-vel-plugin="CurvedNavPlugin"
  120. data-vel-view="item"
  121. >
  122. F
  123. </div>
  124. <div
  125. class="nav-item"
  126. data-vel-plugin="CurvedNavPlugin"
  127. data-vel-view="item"
  128. >
  129. G
  130. </div>
  131. <div
  132. class="nav-item"
  133. data-vel-plugin="CurvedNavPlugin"
  134. data-vel-view="item"
  135. >
  136. H
  137. </div>
  138. <div
  139. class="nav-item"
  140. data-vel-plugin="CurvedNavPlugin"
  141. data-vel-view="item"
  142. >
  143. I
  144. </div>
  145. <div
  146. class="nav-item"
  147. data-vel-plugin="CurvedNavPlugin"
  148. data-vel-view="item"
  149. >
  150. J
  151. </div>
  152. </div>
  153. </div>
  154. </div>
  155. </div>

To add your own content and navigation items, modify the HTML structure within the content-wrapper and nav-container sections. Adjust the text, images, or other elements to suit your website’s needs.

3. Apply basic styling to create a visually appealing layout. Customize the colors, sizes, and other styling properties as needed. The following CSS code includes styles for the curved effect and responsiveness.

  1. * {
  2. box-sizing: border-box;
  3. }
  4. body {
  5. margin: 0;
  6. padding: 0;
  7. display: flex;
  8. align-items: center;
  9. justify-content: center;
  10. width: 100%;
  11. height: 100vh;
  12. background: whitesmoke;
  13. font-family: system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto,
  14. Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;
  15. overflow: hidden;
  16. max-width: 100%;
  17. }
  18.  
  19. .curved-nav {
  20. width: 100%;
  21. height: 100%;
  22. position: relative;
  23. will-change: transform, opacity;
  24. }
  25.  
  26. .curved-nav::before {
  27. position: absolute;
  28. content: '';
  29. width: 100%;
  30. aspect-ratio: 1;
  31. border-radius: 20rem;
  32. background: radial-gradient(
  33. 63.62% 69.52% at 100% 0%,
  34. rgba(247, 214, 98, 0.8) 0%,
  35. rgba(247, 214, 98, 0.168) 52.08%,
  36. rgba(247, 214, 98, 0) 100%
  37. ),
  38. linear-gradient(
  39. 208.42deg,
  40. #f0422a 7.46%,
  41. rgba(240, 88, 42, 0.18) 42.58%,
  42. rgba(240, 101, 42, 0) 64.13%
  43. ),
  44. radial-gradient(
  45. 114.51% 122.83% at 0% -15.36%,
  46. #e74f6a 0%,
  47. rgba(231, 79, 106, 0.22) 66.72%,
  48. rgba(231, 79, 106, 0) 100%
  49. ),
  50. linear-gradient(
  51. 333.95deg,
  52. rgba(83, 208, 236, 0.85) -7.76%,
  53. rgba(83, 208, 236, 0.204) 19.67%,
  54. rgba(138, 137, 190, 0) 35.42%
  55. ),
  56. radial-gradient(
  57. 109.15% 148.57% at 4.46% 98.44%,
  58. #1b3180 0%,
  59. rgba(27, 49, 128, 0) 100%
  60. ),
  61. linear-gradient(141.57deg, #4eadeb 19.08%, rgba(78, 173, 235, 0) 98.72%);
  62. background-blend-mode: normal, normal, normal, normal, multiply, normal;
  63. filter: blur(84px);
  64. will-change: transform;
  65. backface-visibility: hidden;
  66. transform: translate3d(0, 0, 0);
  67. }
  68.  
  69. .nav-container {
  70. display: flex;
  71. width: 100%;
  72. max-width: 375px;
  73. touch-action: none;
  74. will-change: transform, opacity;
  75. }
  76.  
  77. .nav-container-wrapper {
  78. overflow: hidden;
  79. padding: 40px 50px;
  80. }
  81.  
  82. .nav-items {
  83. display: flex;
  84. width: 100%;
  85. --item-size: calc(375px / 5);
  86. }
  87.  
  88. .nav-item {
  89. width: var(--item-size);
  90. height: var(--item-size);
  91. border-radius: 10px;
  92. background: white;
  93. display: flex;
  94. align-items: center;
  95. justify-content: center;
  96. font-size: 24px;
  97. cursor: pointer;
  98. color: #222;
  99. user-select: none;
  100. -webkit-user-select: none;
  101. flex-shrink: 0;
  102. will-change: transform, opacity;
  103. touch-action: none;
  104. }
  105.  
  106. .curved-nav {
  107. width: 100%;
  108. max-width: 375px;
  109. margin: 0 auto;
  110. display: flex;
  111. flex-direction: column;
  112. align-items: center;
  113. justify-content: center;
  114. }
  115.  
  116. .content-wrapper {
  117. border: 1px solid rgba(0, 0, 0, 0.1);
  118. border-radius: 10px;
  119. width: 250px;
  120. height: 250px;
  121. display: flex;
  122. align-items: center;
  123. justify-content: center;
  124. will-change: transform, opacity;
  125. }
  126.  
  127. .content-item {
  128. color: white;
  129. font-size: 300px;
  130. line-height: 0.7;
  131. position: absolute;
  132. will-change: opacity;
  133. }
  134.  
  135. @media (max-width: 500px) {
  136. .nav-items {
  137. --item-size: calc(375px / 6);
  138. }
  139. .curved-nav {
  140. overflow: hidden;
  141. }
  142. }

4. Finally, add the following JavaScript function between <script> tag or external JS file. This code leverages the Veloxi library to create the animated curved navigation. Ensure the Veloxi library is properly loaded in your HTML file.

  1. // Built with Veloxi: https://veloxijs.com/
  2. // Inspired by: https://twitter.com/huseyingayiran/status/1711294397009080397
  3.  
  4. const {
  5. EventBus,
  6. Events,
  7. Plugin,
  8. createApp,
  9. DragEventPlugin,
  10. DragEvent
  11. } = Veloxi
  12.  
  13. const OFFSET = 10
  14.  
  15. class SetActiveIndexEvent {
  16. index
  17. constructor({ index }) {
  18. this.index = index
  19. }
  20. }
  21.  
  22. class NavItem {
  23. view
  24. index
  25. container
  26. initialized = false
  27.  
  28. constructor(view, index, container) {
  29. this.view = view
  30. this.index = index
  31. this.container = container
  32. this.view.position.animator.set('dynamic', { speed: 5 })
  33. this.view.scale.animator.set('dynamic', { speed: 5 })
  34. }
  35.  
  36. init() {
  37. requestAnimationFrame(() => {
  38. this.enableTransition()
  39. })
  40. this.initialized = true
  41. }
  42.  
  43. enableTransition() {
  44. this.view.styles.transition = '0.2s opacity linear'
  45. }
  46.  
  47. update() {
  48. this.updatePosition()
  49. this.updateOpacity()
  50. this.updateScale()
  51. }
  52.  
  53. updateWithOffset(offset) {
  54. const targetSlot = offset > 0 ? this.nextSlot : this.previousSlot
  55. const percentage = Math.abs(offset / this.container.stepSize)
  56. this.updatePositionWithOffset(offset, targetSlot, percentage)
  57. this.updateOpacityWithPercentage(targetSlot, percentage)
  58. this.updateScaleWithPercentage(targetSlot, percentage)
  59. }
  60.  
  61. updatePositionWithOffset(offset, targetSlot, percentage) {
  62. const x = this.currentPosition.x
  63. let y = this.currentPosition.y
  64. switch (targetSlot) {
  65. case slotIndex.ACTIVE:
  66. y -= 10 * percentage
  67. break
  68. case slotIndex.FIRST_NEXT:
  69. case slotIndex.FIRST_PREVIOUS:
  70. const fromActive = this.slotIndex === slotIndex.ACTIVE
  71. y += 25 * (fromActive ? 1 : -1) * percentage
  72. break
  73. case slotIndex.SECOND_NEXT:
  74. case slotIndex.SECOND_PREVIOUS:
  75. const fromFirst = [
  76. slotIndex.FIRST_NEXT,
  77. slotIndex.FIRST_PREVIOUS
  78. ].includes(this.slotIndex)
  79. y += 40 * percentage * (fromFirst ? 1 : -1)
  80. break
  81. default:
  82. const fromSecond = [
  83. slotIndex.SECOND_NEXT,
  84. slotIndex.SECOND_PREVIOUS
  85. ].includes(this.slotIndex)
  86. y += 40 * percentage * (fromSecond ? 1 : -1)
  87. }
  88. this.view.position.set({ x: x + offset, y })
  89. }
  90.  
  91. updateScaleWithPercentage(targetSlot, percentage) {
  92. let scale = this.currentScale
  93.  
  94. switch (targetSlot) {
  95. case slotIndex.ACTIVE:
  96. scale += 0.1 * percentage
  97. break
  98. case slotIndex.FIRST_NEXT:
  99. case slotIndex.FIRST_PREVIOUS:
  100. const fromActive = this.slotIndex === slotIndex.ACTIVE
  101. if (fromActive) {
  102. scale -= 0.1 * percentage
  103. }
  104. break
  105. }
  106. this.view.scale.set({ x: scale, y: scale })
  107. }
  108.  
  109. updateOpacityWithPercentage(targetSlot, percentage) {
  110. let opacity = this.currentOpacity
  111. switch (targetSlot) {
  112. case slotIndex.ACTIVE:
  113. opacity += 0.2 * percentage
  114. break
  115. case slotIndex.FIRST_PREVIOUS:
  116. case slotIndex.FIRST_NEXT:
  117. const fromActive = this.slotIndex === slotIndex.ACTIVE
  118. opacity += 0.2 * percentage * (fromActive ? -1 : 1)
  119. break
  120. case slotIndex.SECOND_PREVIOUS:
  121. case slotIndex.SECOND_NEXT:
  122. const fromFirst = [
  123. slotIndex.FIRST_NEXT,
  124. slotIndex.FIRST_PREVIOUS
  125. ].includes(this.slotIndex)
  126. if (!fromFirst) {
  127. if (
  128. this.firstItemIndexInStart === this.index ||
  129. this.firstItemIndexInEnd === this.index
  130. ) {
  131. opacity += 0.2 * percentage * (fromFirst ? -1 : 1)
  132. } else {
  133. opacity = 0
  134. }
  135. } else {
  136. opacity += 0.2 * percentage * (fromFirst ? -1 : 1)
  137. }
  138. break
  139. default:
  140. const fromSecond = [
  141. slotIndex.SECOND_NEXT,
  142. slotIndex.SECOND_PREVIOUS
  143. ].includes(this.slotIndex)
  144. if (fromSecond) {
  145. opacity += 0.4 * percentage * (fromSecond ? -1 : 1)
  146. } else {
  147. opacity = 0
  148. }
  149. }
  150. this.view.styles.opacity = `${opacity}`
  151. }
  152.  
  153. updatePosition() {
  154. const shouldAnimate = this.container.shouldAnimateMap.get(this.index)
  155. const x = this.slotPosition.x
  156. let y = this.slotPosition.y
  157. switch (this.slotIndex) {
  158. case slotIndex.ACTIVE:
  159. break
  160. case slotIndex.FIRST_NEXT:
  161. case slotIndex.FIRST_PREVIOUS:
  162. y += 10
  163. break
  164. case slotIndex.SECOND_NEXT:
  165. case slotIndex.SECOND_PREVIOUS:
  166. y += 40
  167. break
  168. default:
  169. y += 100
  170. }
  171. this.view.position.set({ x, y }, shouldAnimate)
  172. }
  173.  
  174. get currentPosition() {
  175. const x = this.slotPosition.x
  176. let y = this.slotPosition.y
  177. switch (this.slotIndex) {
  178. case slotIndex.ACTIVE:
  179. break
  180. case slotIndex.FIRST_NEXT:
  181. case slotIndex.FIRST_PREVIOUS:
  182. y += 10
  183. break
  184. case slotIndex.SECOND_NEXT:
  185. case slotIndex.SECOND_PREVIOUS:
  186. y += 40
  187. break
  188. default:
  189. y += 80
  190. }
  191. return { x, y }
  192. }
  193.  
  194. get currentOpacity() {
  195. let opacity = 0
  196. switch (this.slotIndex) {
  197. case slotIndex.ACTIVE:
  198. opacity = 1
  199. break
  200. case slotIndex.FIRST_PREVIOUS:
  201. case slotIndex.FIRST_NEXT:
  202. opacity = 0.425
  203. break
  204. case slotIndex.SECOND_PREVIOUS:
  205. case slotIndex.SECOND_NEXT:
  206. opacity = 0.2
  207. break
  208. }
  209. return opacity
  210. }
  211.  
  212. get currentScale() {
  213. let scale = 0.75
  214. switch (this.slotIndex) {
  215. case slotIndex.ACTIVE:
  216. scale = 1
  217. break
  218. case slotIndex.FIRST_PREVIOUS:
  219. case slotIndex.FIRST_NEXT:
  220. scale = 0.75
  221. break
  222. case slotIndex.SECOND_PREVIOUS:
  223. case slotIndex.SECOND_NEXT:
  224. scale = 0.75
  225. break
  226. }
  227. return scale
  228. }
  229.  
  230. updateOpacity() {
  231. let opacity = 0
  232. switch (this.slotIndex) {
  233. case slotIndex.ACTIVE:
  234. opacity = 1
  235. break
  236. case slotIndex.FIRST_PREVIOUS:
  237. case slotIndex.FIRST_NEXT:
  238. opacity = 0.425
  239. break
  240. case slotIndex.SECOND_PREVIOUS:
  241. case slotIndex.SECOND_NEXT:
  242. opacity = 0.2
  243. break
  244. }
  245.  
  246. this.view.styles.opacity = `${opacity}`
  247. }
  248.  
  249. updateScale() {
  250. let scale = 0.75
  251. switch (this.slotIndex) {
  252. case slotIndex.ACTIVE:
  253. scale = 1
  254. break
  255. case slotIndex.FIRST_PREVIOUS:
  256. case slotIndex.FIRST_NEXT:
  257. scale = 0.75
  258. break
  259. case slotIndex.SECOND_PREVIOUS:
  260. case slotIndex.SECOND_NEXT:
  261. scale = 0.75
  262. break
  263. }
  264. this.view.scale.set({ x: scale, y: scale }, this.initialized)
  265. }
  266.  
  267. get nextSlot() {
  268. return wrapAround(this.slotIndex, Object.keys(slotIndex).length, 1)
  269. }
  270.  
  271. get previousSlot() {
  272. return wrapAround(this.slotIndex, Object.keys(slotIndex).length, -1)
  273. }
  274.  
  275. get slotPosition() {
  276. return this.container.getSlotPositionForItemIndex(this.index)
  277. }
  278.  
  279. get activeIndex() {
  280. return this.container.activeIndex
  281. }
  282.  
  283. get slotIndex() {
  284. return this.container.getSlotForIndex(this.index)
  285. }
  286.  
  287. getItemIndexForSlot(slot) {
  288. return this.container.getItemIndeciesForSlot(slot)[0]
  289. }
  290.  
  291. get firstItemIndexInStart() {
  292. const secondPreviousIndex = this.getItemIndexForSlot(
  293. slotIndex.SECOND_PREVIOUS
  294. )
  295. return wrapAround(secondPreviousIndex, this.container.totalItems, -1)
  296. }
  297.  
  298. get firstItemIndexInEnd() {
  299. const secondNextItemIndex = this.getItemIndexForSlot(
  300. slotIndex.SECOND_NEXT
  301. )
  302. return wrapAround(secondNextItemIndex, this.container.totalItems, 1)
  303. }
  304. }
  305.  
  306. const slotIndex = {
  307. START: 0,
  308. SECOND_PREVIOUS: 1,
  309. FIRST_PREVIOUS: 2,
  310. ACTIVE: 3,
  311. FIRST_NEXT: 4,
  312. SECOND_NEXT: 5,
  313. END: 6
  314. }
  315.  
  316. function wrapAround(current, total, amount) {
  317. return (current + total + amount) % total
  318. }
  319.  
  320. function flipMap(map) {
  321. const flippedMap = new Map()
  322.  
  323. for (const [key, value] of map) {
  324. if (!flippedMap.has(value)) {
  325. flippedMap.set(value, [key])
  326. } else {
  327. flippedMap.get(value).push(key)
  328. }
  329. }
  330.  
  331. return flippedMap
  332. }
  333.  
  334. class NavContainer {
  335. plugin
  336. view
  337. activeIndex
  338. shouldAnimateMap = new Map()
  339.  
  340. _itemIndexSlotMap = new Map()
  341. _slotItemIndexMap = new Map()
  342.  
  343. constructor(plugin, view) {
  344. this.plugin = plugin
  345. this.view = view
  346. this.activeIndex = this.view.data.activeIndex
  347. ? parseInt(this.view.data.activeIndex)
  348. : 0
  349.  
  350. for (let index = 0; index < this.totalItems; index++) {
  351. this.shouldAnimateMap.set(index, false)
  352. }
  353. }
  354.  
  355. get stepSize() {
  356. return this.plugin.stepSize
  357. }
  358.  
  359. get itemSize() {
  360. return this.plugin.itemSize
  361. }
  362.  
  363. get totalItems() {
  364. return this.plugin.totalItems
  365. }
  366.  
  367. updateWithOffset(offset) {
  368. const steps = Math.floor(Math.abs(offset / this.stepSize))
  369. const queue = []
  370. let currentIndex = this.activeIndex
  371. for (let step = 0; step < steps; step++) {
  372. const stepDirection = offset < 0 ? 1 : -1
  373. const itemIndex = wrapAround(
  374. currentIndex,
  375. this.totalItems,
  376. stepDirection
  377. )
  378. queue.push(itemIndex)
  379. currentIndex = itemIndex
  380. }
  381. queue.forEach((itemIndex, index) => {
  382. setTimeout(() => {
  383. this.plugin.setActiveIndex(itemIndex)
  384. }, 100 * index)
  385. })
  386. }
  387.  
  388. setActiveIndex(newActiveIndex) {
  389. const previousItemIndexSlot = this.itemIndexSlotMap
  390. this.activeIndex = newActiveIndex
  391. this.setItemIndexSlotMap()
  392. const newItemIndexSlot = this.itemIndexSlotMap
  393.  
  394. const visibleSlots = [
  395. slotIndex.ACTIVE,
  396. slotIndex.FIRST_PREVIOUS,
  397. slotIndex.SECOND_PREVIOUS,
  398. slotIndex.FIRST_NEXT,
  399. slotIndex.SECOND_NEXT
  400. ]
  401. for (let index = 0; index < this.totalItems; index++) {
  402. const shouldAnimate =
  403. visibleSlots.includes(previousItemIndexSlot.get(index)) ||
  404. visibleSlots.includes(newItemIndexSlot.get(index))
  405. this.shouldAnimateMap.set(index, shouldAnimate)
  406. }
  407. }
  408.  
  409. get slotItemIndexMap() {
  410. return this._slotItemIndexMap
  411. }
  412.  
  413. get itemIndexSlotMap() {
  414. return this._itemIndexSlotMap
  415. }
  416.  
  417. getSlotForIndex(itemIndex) {
  418. return this.itemIndexSlotMap.get(itemIndex)
  419. }
  420.  
  421. getItemIndeciesForSlot(slot) {
  422. return this.slotItemIndexMap.get(slot)
  423. }
  424.  
  425. setItemIndexSlotMap() {
  426. this._itemIndexSlotMap.clear()
  427. const activeIndex = this.activeIndex
  428.  
  429. const firstPreviousIndex = wrapAround(activeIndex, this.totalItems, -1)
  430. const secondPreviousIndex = wrapAround(activeIndex, this.totalItems, -2)
  431. const firstNextIndex = wrapAround(activeIndex, this.totalItems, 1)
  432. const secondNextIndex = wrapAround(activeIndex, this.totalItems, 2)
  433.  
  434. for (let index = 0; index < this.totalItems; index++) {
  435. if (index === activeIndex) {
  436. this._itemIndexSlotMap.set(index, slotIndex.ACTIVE)
  437. continue
  438. }
  439.  
  440. if (index === firstPreviousIndex) {
  441. this._itemIndexSlotMap.set(index, slotIndex.FIRST_PREVIOUS)
  442. continue
  443. }
  444.  
  445. if (index === secondPreviousIndex) {
  446. this._itemIndexSlotMap.set(index, slotIndex.SECOND_PREVIOUS)
  447. continue
  448. }
  449.  
  450. if (index === firstNextIndex) {
  451. this._itemIndexSlotMap.set(index, slotIndex.FIRST_NEXT)
  452. continue
  453. }
  454.  
  455. if (index === secondNextIndex) {
  456. this._itemIndexSlotMap.set(index, slotIndex.SECOND_NEXT)
  457. continue
  458. }
  459.  
  460. if (index === wrapAround(secondNextIndex, this.totalItems, 1)) {
  461. this._itemIndexSlotMap.set(index, slotIndex.END)
  462. continue
  463. }
  464.  
  465. if (index === wrapAround(secondNextIndex, this.totalItems, 2)) {
  466. this._itemIndexSlotMap.set(index, slotIndex.END)
  467. continue
  468. }
  469.  
  470. if (index === wrapAround(secondPreviousIndex, this.totalItems, -1)) {
  471. this._itemIndexSlotMap.set(index, slotIndex.START)
  472. continue
  473. }
  474.  
  475. if (index === wrapAround(secondPreviousIndex, this.totalItems, -2)) {
  476. this._itemIndexSlotMap.set(index, slotIndex.START)
  477. continue
  478. }
  479.  
  480. if (index > activeIndex) {
  481. this._itemIndexSlotMap.set(index, slotIndex.END)
  482. continue
  483. }
  484.  
  485. if (index < activeIndex) {
  486. this._itemIndexSlotMap.set(index, slotIndex.START)
  487. continue
  488. }
  489. }
  490. this._slotItemIndexMap = flipMap(this.itemIndexSlotMap)
  491. }
  492.  
  493. getSlotPositionForItemIndex(index) {
  494. const slot = this.itemIndexSlotMap.get(index)
  495. return this.slotPositions[slot]
  496. }
  497.  
  498. get indicatorPosition() {
  499. return {
  500. x: this.view.position.x + this.view.size.width / 2,
  501. y: this.view.position.y + this.view.size.height / 2
  502. }
  503. }
  504.  
  505. get slotPositions() {
  506. const result = []
  507. // Active Slot
  508. result[slotIndex.ACTIVE] = {
  509. x: this.indicatorPosition.x - this.itemSize / 2,
  510. y: this.indicatorPosition.y - this.itemSize / 2
  511. }
  512.  
  513. result[slotIndex.FIRST_PREVIOUS] = {
  514. x: result[slotIndex.ACTIVE].x - OFFSET - this.itemSize,
  515. y: result[slotIndex.ACTIVE].y
  516. }
  517.  
  518. result[slotIndex.SECOND_PREVIOUS] = {
  519. x: result[slotIndex.FIRST_PREVIOUS].x - this.itemSize,
  520. y: result[slotIndex.FIRST_PREVIOUS].y
  521. }
  522.  
  523. result[slotIndex.START] = {
  524. x: result[slotIndex.SECOND_PREVIOUS].x - this.itemSize,
  525. y: result[slotIndex.SECOND_PREVIOUS].y
  526. }
  527.  
  528. result[slotIndex.FIRST_NEXT] = {
  529. x: result[slotIndex.ACTIVE].x + OFFSET + this.itemSize,
  530. y: result[slotIndex.ACTIVE].y
  531. }
  532.  
  533. result[slotIndex.SECOND_NEXT] = {
  534. x: result[slotIndex.FIRST_NEXT].x + this.itemSize,
  535. y: result[slotIndex.FIRST_NEXT].y
  536. }
  537.  
  538. result[slotIndex.END] = {
  539. x: result[slotIndex.SECOND_NEXT].x + this.itemSize,
  540. y: result[slotIndex.SECOND_NEXT].y
  541. }
  542.  
  543. return result
  544. }
  545. }
  546.  
  547. class ContentItem {
  548. view
  549. index
  550. navItemsContainer
  551.  
  552. constructor(view, index, navItemsContainer) {
  553. this.view = view
  554. this.index = index
  555. this.navItemsContainer = navItemsContainer
  556. this.init()
  557. this.update()
  558. }
  559.  
  560. init() {
  561. requestAnimationFrame(() => {
  562. this.enableTransition()
  563. })
  564. }
  565.  
  566. enableTransition() {
  567. this.view.styles.transition = '0.2s ease-in-out opacity'
  568. }
  569.  
  570. update() {
  571. if (this.index !== this.activeIndex) {
  572. this.hide()
  573. } else {
  574. this.show()
  575. }
  576. }
  577.  
  578. hide() {
  579. this.view.styles.opacity = '0'
  580. }
  581.  
  582. show() {
  583. this.view.styles.opacity = '1'
  584. }
  585.  
  586. get activeIndex() {
  587. return this.navItemsContainer.activeIndex
  588. }
  589. }
  590.  
  591. class CurvedNavPlugin extends Plugin {
  592. static pluginName = 'CurvedNavPlugin'
  593.  
  594. items
  595. itemsContainer
  596.  
  597. contentItems
  598.  
  599. lastDragOffset = 0
  600. isDragging = false
  601.  
  602. dragEventPlugin = this.useEventPlugin(DragEventPlugin)
  603.  
  604. setup() {
  605. const itemViews = this.getViews('item')
  606. const itemsContainerView = this.getView('itemsContainer')
  607. this.itemsContainer = new NavContainer(this, itemsContainerView)
  608.  
  609. this.dragEventPlugin.addView(itemsContainerView)
  610. this.dragEventPlugin.on(DragEvent, this.onDrag.bind(this))
  611.  
  612. this.items = itemViews.map(
  613. (view, index) => new NavItem(view, index, this.itemsContainer)
  614. )
  615.  
  616. this.itemsContainer.setItemIndexSlotMap()
  617.  
  618. this.items.forEach((item) => {
  619. item.update()
  620. item.init()
  621. })
  622.  
  623. this.contentItems = this.getViews('contentItem').map(
  624. (view, index) => new ContentItem(view, index, this.itemsContainer)
  625. )
  626. }
  627.  
  628. onDrag(event) {
  629. if (event.isDragging) {
  630. this.isDragging = true
  631. } else {
  632. requestAnimationFrame(() => {
  633. this.isDragging = false
  634. })
  635. }
  636. if (event.isDragging) {
  637. const diff = Math.abs(event.previousX - event.x)
  638. const damping = diff > 50 ? 0.2 : 1
  639. const offset = damping * (event.width + this.lastDragOffset * -1)
  640. if (Math.abs(offset) >= this.stepSize) {
  641. this.lastDragOffset = event.width
  642. }
  643. this.itemsContainer.updateWithOffset(offset)
  644. this.items.forEach((item) => {
  645. item.updateWithOffset(offset)
  646. })
  647. } else {
  648. this.lastDragOffset = 0
  649. this.items.forEach((item) => {
  650. item.updateWithOffset(0)
  651. })
  652. }
  653. }
  654.  
  655. onDataChanged(data) {
  656. if (data.dataName === 'activeIndex') {
  657. const activeIndex = parseInt(data.dataValue)
  658. this.itemsContainer.setActiveIndex(activeIndex)
  659. this.items.forEach((item) => item.update())
  660. this.contentItems.forEach((item) => item.update())
  661. }
  662. }
  663.  
  664. subscribeToEvents(eventBus) {
  665. eventBus.subscribeToEvent(Events.PointerClickEvent, ({ target }) => {
  666. if (this.isDragging) return
  667. this.items.forEach((item, index) => {
  668. if (target === item.view.element) {
  669. this.setActiveIndex(index)
  670. }
  671. })
  672. })
  673. }
  674.  
  675. setActiveIndex(index) {
  676. this.emit(SetActiveIndexEvent, { index })
  677. }
  678.  
  679. get totalItems() {
  680. return this.getViews('item').length
  681. }
  682.  
  683. get itemSize() {
  684. return this.items[0].view.size.width
  685. }
  686.  
  687. get stepSize() {
  688. return this.itemSize + OFFSET
  689. }
  690. }
  691.  
  692. const app = createApp()
  693. app.addPlugin(CurvedNavPlugin)
  694. app.run()
  695.  
  696. const containerNav = document.querySelector(
  697. '[data-vel-view="itemsContainer"]'
  698. )
  699. app.onPluginEvent(CurvedNavPlugin, SetActiveIndexEvent, ({ index }) => {
  700. containerNav.dataset.velDataActiveIndex = `${index}`
  701. })

That’s all! hopefully, you have successfully created an animated curved nav using JavaScript. If you have any questions or suggestions, feel free to comment below.

Leave a Comment

This site uses Akismet to reduce spam. Learn how your comment data is processed.

About CodeHim

Free Web Design Code & Scripts - CodeHim is one of the BEST developer websites that provide web designers and developers with a simple way to preview and download a variety of free code & scripts. All codes published on CodeHim are open source, distributed under OSD-compliant license which grants all the rights to use, study, change and share the software in modified and unmodified form. Before publishing, we test and review each code snippet to avoid errors, but we cannot warrant the full correctness of all content. All trademarks, trade names, logos, and icons are the property of their respective owners... find out more...

Please Rel0ad/PressF5 this page if you can't click the download/preview link

X