Skip to content

CarelessCourage/Moonbow

Folders and files

NameName
Last commit message
Last commit date

Latest commit

ย 

History

80 Commits
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 

Repository files navigation

chrome-capture-2023-1-5 (1)

Moonbow ๐ŸŒš+๐ŸŒˆ

Vue img component for adding GLSL to images ๐Ÿ”ฅ๐Ÿ”ฅ๐Ÿ”ฅ

Test it out yourself. Head over to Moonbow to get an idea of whats possible and feel out the performance. Or jump right into a code example with this StackBlitz.

โš—๏ธ How it works

Moonbow leverages three.js to create a 3D space in webGL. It creates a 3 dimensional plane for each image and sticks it to the size and position of the proxy HTML img element. It re-attatches this plane to the img element on every animation frame to keep it consistent with the layout. But it saves on performance by only attaching the planes that are inside the viewport. It does this by using an intersection observer to check for images in view.

  • ๐Ÿ˜ฝ Simple - Just a simple image component needed. Nothing more
  • ๐Ÿ’ช Flexible - Flexible primitives underneath that let you build your own logic
  • ๐Ÿ”ญ Typesafe - Written fully in typescript
  • ๐Ÿ› ๏ธ Maintainable - HTML stays descriptive of content letting canvas images flows with the HTML elements
  • ๐Ÿ‘จโ€๐Ÿฆฝ Accessible - Since canvas elements have their HTML counterparts you dont lose accessibility controls like other approaches would

๐Ÿงช Benefits of this approach

This apprach lets you take advantage of GLSL for your images while keeping the DOM descriptive of your content. Images get created in webGL with a HTML proxy element in the DOM taking up space and flowing with your layout. This is also great for accessibility since it means we can accomplish complex image manipulation without sacrificing on the browser inbuilt accessibility tools.

๐Ÿ“œ Resources

Learn how to write GLSL: Book of Shaders

๐Ÿ“ฆ Installation

npm install moonbow

๐Ÿ—๏ธ Setup

You can inject GLSL as a string but if you want to actually use it you're going to want to store GLSL in files that you can import. In order to do that you need to let Vite know how to handle GLSL files though. Import the vite-plugin-glsl package and register it as a plugin in your vite.config file.

import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'
import glsl from 'vite-plugin-glsl'

export default defineConfig({
  plugins: [vue(), glsl()],
})

๐Ÿ”ฎ Usage

Simple example using default GLSL. Here's a StackBlitz if you just want to jump right into it as fast as possible.

<script setup>
import { Moon } from "moonbow"
import "moonbow/dist/style.css"
const imageURL = "https://images.unsplash.com/photo-1642059893618"
</script>

<template>
  <Moon :src="imageURL"/>
</template>

Adding your own custom GLSL

<script setup>
import { Moon } from "moonbow"
import "moonbow/dist/style.css"

import vertexShader from '../shaders/scrollDeform/vertex.glsl'
import fragmentShader from '../shaders/scrollDeform/fragment.glsl'
</script>

<template>
  <Moon 
    src="https://images.unsplash.com/photo-1642059893618" 
    :vertexShader="vertexShader"
    :fragmentShader="fragmentShader"
  />
</template>

Adding custom uniforms and changing uniforms dynamically from JavaScript

<script setup>
import { watch } from 'vue'
import { Moon, useScroll } from "moonbow"
import "moonbow/dist/style.css"

import vertexShader from '../shaders/scrollDeform/vertex.glsl'
import fragmentShader from '../shaders/scrollDeform/fragment.glsl'

let uniforms = {
  uVelocity: { value: 0 },
}

const velocity = useScroll()
const uniformControls = {
  vertexShader,
  fragmentShader,
  uniforms,
  uniformAction: (material) => {
    watch(velocity, (velocity) => {
      material.uniforms.uVelocity.value = velocity
    })
  }
}
</script>

<template>
  <Moon
    src="https://images.unsplash.com/photo-1642059893618"
    v-bind="uniformControls"
  />
</template>

Most of the time you will probably want more than one image. And these multiple images probably will use the same effect, sometimes maybe even in different files. Setting the same shader object over and over can be cumbersome. So I've provided a function for setting the default shader for all images. Just set this once and never worry about setting it again unless you want to overwrite the defaults on a specific image.

<script setup>
import { watch } from 'vue'
import { Moon, useScroll, defaultGLSL } from "moonbow"
import "moonbow/dist/style.css"

import vertexShader from '../shaders/scrollDeform/vertex.glsl'
import fragmentShader from '../shaders/scrollDeform/fragment.glsl'

let uniforms = {
  uVelocity: { value: 0 },
}

const scroll = useScroll()
function uniformAction(material: any) {
  watch(scroll, (s) => {
    material.uniforms.uVelocity.value = s
  })
}

defaultGLSL({
  uniforms,
  vertexShader,
  fragmentShader,
  uniformAction
})
</script>

<template>
  <Moon src="https://images.unsplash.com/photo-1642059893618"/>
  <Moon src="https://images.unsplash.com/photo-1642059893618"/>
  <Moon src="https://images.unsplash.com/photo-1642059893618"/>
  <Moon src="https://images.unsplash.com/photo-1642059893618"/>
  <Moon src="https://images.unsplash.com/photo-1642059893618"/>
</template>

๐Ÿงฌ Primitives

Moonbow exposes the underlying primitives developed to make the Moon image component. You can easily use these primitives to build your own solution. Below is the entire code for the Moon component showcasing the way it uses the primities.

<script setup lang="ts">
import { ref, onMounted  } from "vue"
import { useShader, scene, useImage, syncProxyHTML } from "moonbow";

const props = defineProps({
  src: string;
  vertexShader?: string;
  fragmentShader?: string;
  uniforms?: any;
  uniformAction?: (material: ShaderType) => void;
})

const imageRef = ref(null)

onMounted(() => {
  const src = props.src;
  const element = imageRef.value
  const scene = sceneRef.value
  const shader: MoonbowShader = {
    vertexShader: props.vertexShader,
    fragmentShader: props.fragmentShader,
    uniforms: props.uniforms,
    uniformAction: props.uniformAction
  }

  const material = useShader(element, shader)
  const proxy = useImage({scene, element, material})
  syncProxyHTML({ proxy, src })
})
</script>

<template>
  <div class="img-wrapper">
    <img :src="src" ref="imageRef"/>
  </div>
</template>

Postprocessing function to let you apply GLSL to all elements uniformly. Example with plain GLSL without dynamic uniforms

<script setup>
import { postProcessing } from '../composables/canvas'

import vertexShader from '../shaders/bottomScale/vertex.glsl';
import fragmentShader from '../shaders/bottomScale/fragment.glsl';

const uniforms = {
  uExample: { value: 0 },
}

postProcessing({
  uniforms,
  vertexShader,
  fragmentShader,
})
</script>

Postprocessing function with uniformAction/dynamic uniforms example.

<script setup>
import { postProcessing } from '../composables/canvas'

import vertexShader from '../shaders/bottomScale/vertex.glsl';
import fragmentShader from '../shaders/bottomScale/fragment.glsl';

const uniforms = {
  uExample: { value: 0 },
}

const shader = {
  uniforms,
  vertexShader,
  fragmentShader,
  uniformAction: (m) => {
    watch(velocity, (velocity) => {
      m.uniforms.uVelocity.value = velocity
    })
  }
}

postProcessing(shader)
</script>