forked from external-repos/noisedash
Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e961568178 |
@@ -1,2 +0,0 @@
|
|||||||
CHOKIDAR_USEPOLLING=true
|
|
||||||
WDS_SOCKET_PORT=0
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "Noisedash Dev",
|
|
||||||
"image": "node:18",
|
|
||||||
"forwardPorts": [8080, 1432],
|
|
||||||
"portsAttributes": {
|
|
||||||
"8080": { "label": "Vue dev server" },
|
|
||||||
"1432": { "label": "Noisedash API" }
|
|
||||||
},
|
|
||||||
"postCreateCommand": "npm install",
|
|
||||||
"remoteEnv": {
|
|
||||||
"HOST": "0.0.0.0"
|
|
||||||
},
|
|
||||||
"containerEnv": {
|
|
||||||
"HOST": "0.0.0.0"
|
|
||||||
},
|
|
||||||
"runArgs": ["--env-file", ".devcontainer/devcontainer.env"],
|
|
||||||
"remoteUser": "node",
|
|
||||||
"customizations": {
|
|
||||||
"vscode": {
|
|
||||||
"settings": {
|
|
||||||
"editor.defaultFormatter": "esbenp.prettier-vscode",
|
|
||||||
"eslint.format.enable": true,
|
|
||||||
"eslint.validate": ["javascript", "vue"],
|
|
||||||
"terminal.integrated.defaultProfile.linux": "bash",
|
|
||||||
"vetur.validation.template": true,
|
|
||||||
"vetur.validation.script": true,
|
|
||||||
"vetur.validation.style": true
|
|
||||||
},
|
|
||||||
"extensions": [
|
|
||||||
"dbaeumer.vscode-eslint",
|
|
||||||
"esbenp.prettier-vscode",
|
|
||||||
"octref.vetur"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"updateContentCommand": "",
|
|
||||||
"onCreateCommand": "mkdir -p samples sessions log",
|
|
||||||
"postStartCommand": "npm run server & npm run serve"
|
|
||||||
}
|
|
||||||
3
.gitattributes
vendored
3
.gitattributes
vendored
@@ -1,3 +0,0 @@
|
|||||||
* text=auto eol=lf
|
|
||||||
*.{cmd,[cC][mM][dD]} text eol=crlf
|
|
||||||
*.{bat,[bB][aA][tT]} text eol=crlf
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
Gitea Actions for Docker builds
|
|
||||||
|
|
||||||
This folder contains workflows for building and pushing Docker images to the Gitea Container Registry.
|
|
||||||
|
|
||||||
Setup
|
|
||||||
- Ensure Gitea Actions is enabled in your instance and for this repo.
|
|
||||||
- Create the following repository secrets:
|
|
||||||
- REGISTRY_HOST: your Gitea host (e.g., gitea.example.com)
|
|
||||||
- REGISTRY_OWNER: your namespace/user/org (e.g., ryan)
|
|
||||||
- REGISTRY_REPO: repository name (e.g., noisedash)
|
|
||||||
- REGISTRY_USERNAME: registry username
|
|
||||||
- REGISTRY_PASSWORD: registry password or a scoped token
|
|
||||||
|
|
||||||
Notes
|
|
||||||
- The workflow builds on pushes to main/master/dev and on tags v*.*.*.
|
|
||||||
- It produces multi-arch images (amd64, arm64). Adjust platforms if not needed.
|
|
||||||
- Image name resolves to: ${REGISTRY_HOST}/${REGISTRY_OWNER}/${REGISTRY_REPO} with tags derived from branch/tag/sha.
|
|
||||||
@@ -1,62 +0,0 @@
|
|||||||
name: build-and-push-docker
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches: [ main, master, dev ]
|
|
||||||
tags:
|
|
||||||
- 'v*.*.*'
|
|
||||||
pull_request:
|
|
||||||
branches: [ main, master, dev ]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
docker:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Install system Dependencies
|
|
||||||
run: |
|
|
||||||
apt-get update && apt-get install -y curl jq docker.io
|
|
||||||
|
|
||||||
- name: Set up Docker Buildx
|
|
||||||
uses: docker/setup-buildx-action@v3
|
|
||||||
|
|
||||||
- name: Log in to Gitea Container Registry
|
|
||||||
uses: docker/login-action@v3
|
|
||||||
with:
|
|
||||||
registry: gitea.purpleraft.com
|
|
||||||
username: ${{ gitea.actor }}
|
|
||||||
password: ${{ secrets.REGISTRY_TOKEN }}
|
|
||||||
|
|
||||||
- name: Extract metadata (tags, labels)
|
|
||||||
id: meta
|
|
||||||
uses: docker/metadata-action@v5
|
|
||||||
with:
|
|
||||||
images: gitea.purpleraft.com/${{ gitea.repository_owner }}/noisedash
|
|
||||||
tags: |
|
|
||||||
type=ref,event=branch
|
|
||||||
type=ref,event=tag
|
|
||||||
type=sha
|
|
||||||
type=raw,value=latest,enable={{is_default_branch}}
|
|
||||||
|
|
||||||
- name: Build (PR only)
|
|
||||||
if: ${{ gitea.event_name == 'pull_request' }}
|
|
||||||
uses: docker/build-push-action@v6
|
|
||||||
with:
|
|
||||||
context: .
|
|
||||||
file: ./Dockerfile
|
|
||||||
push: false
|
|
||||||
platforms: linux/amd64
|
|
||||||
labels: ${{ steps.meta.outputs.labels }}
|
|
||||||
|
|
||||||
- name: Build and push (branches/tags)
|
|
||||||
if: ${{ gitea.event_name != 'pull_request' }}
|
|
||||||
uses: docker/build-push-action@v6
|
|
||||||
with:
|
|
||||||
context: .
|
|
||||||
file: ./Dockerfile
|
|
||||||
push: true
|
|
||||||
platforms: linux/amd64
|
|
||||||
tags: ${{ steps.meta.outputs.tags }}
|
|
||||||
labels: ${{ steps.meta.outputs.labels }}
|
|
||||||
16
Dockerfile
16
Dockerfile
@@ -1,24 +1,10 @@
|
|||||||
# NoiseDash - Audio sample management platform
|
|
||||||
FROM node:20
|
FROM node:20
|
||||||
|
|
||||||
LABEL maintainer="kaythomas@pm.me"
|
LABEL maintainer="kaythomas@pm.me"
|
||||||
|
|
||||||
# Set working directory
|
|
||||||
WORKDIR /var/noisedash
|
WORKDIR /var/noisedash
|
||||||
|
|
||||||
# Copy package files and install dependencies
|
|
||||||
COPY package*.json ./
|
COPY package*.json ./
|
||||||
RUN npm install --force
|
RUN npm install --force
|
||||||
|
|
||||||
# Copy application code (includes default config files)
|
|
||||||
COPY . .
|
COPY . .
|
||||||
|
ENV NODE_ENV production
|
||||||
# Set production environment and build frontend
|
|
||||||
ENV NODE_ENV=production
|
|
||||||
RUN npm run build
|
RUN npm run build
|
||||||
|
|
||||||
# Expose application port
|
|
||||||
EXPOSE 1432
|
EXPOSE 1432
|
||||||
|
|
||||||
# Start the server
|
|
||||||
CMD [ "node", "server/bin/www.js" ]
|
CMD [ "node", "server/bin/www.js" ]
|
||||||
|
|||||||
@@ -2,13 +2,16 @@ version: "3"
|
|||||||
|
|
||||||
services:
|
services:
|
||||||
noisedash:
|
noisedash:
|
||||||
image: gitea.purpleraft.com/ryan/noisedash:latest
|
image: noisedash/noisedash:latest
|
||||||
|
container_name: noisedash
|
||||||
ports:
|
ports:
|
||||||
- "1432:1432"
|
- "1432:1432"
|
||||||
volumes:
|
volumes:
|
||||||
- ./db:/var/noisedash/db
|
- db:/var/noisedash/db
|
||||||
- ./samples:/var/noisedash/samples
|
- samples:/var/noisedash/samples
|
||||||
# Optional: Override default config if you have custom settings
|
- ./config/default.json:/var/noisedash/config/default.json
|
||||||
# - ./config/default.json:/var/noisedash/config/default.json
|
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
db:
|
||||||
|
samples:
|
||||||
|
|
||||||
|
|||||||
3566
package-lock.json
generated
3566
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -33,7 +33,6 @@
|
|||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/core": "^7.12.16",
|
"@babel/core": "^7.12.16",
|
||||||
"@babel/eslint-parser": "^7.12.16",
|
"@babel/eslint-parser": "^7.12.16",
|
||||||
"@vue/cli-plugin-pwa": "^5.0.8",
|
|
||||||
"@vue/cli-plugin-babel": "^5.0.8",
|
"@vue/cli-plugin-babel": "^5.0.8",
|
||||||
"@vue/cli-plugin-eslint": "^5.0.8",
|
"@vue/cli-plugin-eslint": "^5.0.8",
|
||||||
"@vue/cli-plugin-router": "^5.0.8",
|
"@vue/cli-plugin-router": "^5.0.8",
|
||||||
|
|||||||
@@ -5,8 +5,6 @@
|
|||||||
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||||
<meta name="viewport" content="width=device-width,initial-scale=1.0">
|
<meta name="viewport" content="width=device-width,initial-scale=1.0">
|
||||||
<link rel="icon" href="<%= BASE_URL %>favicon.ico">
|
<link rel="icon" href="<%= BASE_URL %>favicon.ico">
|
||||||
<meta name="theme-color" content="#121212">
|
|
||||||
<link rel="manifest" href="<%= BASE_URL %>manifest.webmanifest">
|
|
||||||
<title><%= htmlWebpackPlugin.options.title %></title>
|
<title><%= htmlWebpackPlugin.options.title %></title>
|
||||||
<link rel="stylesheet" href="https://fonts.googleapis.com/css?family=Roboto:100,300,400,500,700,900">
|
<link rel="stylesheet" href="https://fonts.googleapis.com/css?family=Roboto:100,300,400,500,700,900">
|
||||||
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@mdi/font@latest/css/materialdesignicons.min.css">
|
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@mdi/font@latest/css/materialdesignicons.min.css">
|
||||||
|
|||||||
@@ -1,16 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "Noisedash",
|
|
||||||
"short_name": "Noisedash",
|
|
||||||
"start_url": "/",
|
|
||||||
"display": "standalone",
|
|
||||||
"background_color": "#121212",
|
|
||||||
"theme_color": "#121212",
|
|
||||||
"description": "Ambient noise generator",
|
|
||||||
"icons": [
|
|
||||||
{
|
|
||||||
"src": "favicon.ico",
|
|
||||||
"sizes": "64x64 32x32 24x24 16x16",
|
|
||||||
"type": "image/x-icon"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -8,7 +8,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import AppBar from '@/components/AppBar.vue'
|
import AppBar from '@/components/AppBar'
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'App',
|
name: 'App',
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import * as Tone from 'tone'
|
import * as Tone from 'tone'
|
||||||
import { setMediaSessionMetadata } from '@/registerServiceWorker'
|
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'Noise',
|
name: 'Noise',
|
||||||
@@ -242,17 +241,6 @@ export default {
|
|||||||
})
|
})
|
||||||
|
|
||||||
Tone.Transport.start('+0.1')
|
Tone.Transport.start('+0.1')
|
||||||
|
|
||||||
// Update Media Session so playback can continue in background with proper metadata
|
|
||||||
try {
|
|
||||||
setMediaSessionMetadata({ title: (this.selectedProfile && this.selectedProfile.text) ? this.selectedProfile.text : 'Noisedash' })
|
|
||||||
if ('mediaSession' in navigator) {
|
|
||||||
navigator.mediaSession.playbackState = 'playing'
|
|
||||||
navigator.mediaSession.setActionHandler('play', () => this.play())
|
|
||||||
navigator.mediaSession.setActionHandler('pause', () => this.stop())
|
|
||||||
navigator.mediaSession.setActionHandler('stop', () => this.stop())
|
|
||||||
}
|
|
||||||
} catch (e) { /* no-op */ }
|
|
||||||
},
|
},
|
||||||
playSporadicSample (id) {
|
playSporadicSample (id) {
|
||||||
const sample = this.loadedSamples.find(s => s.id === id)
|
const sample = this.loadedSamples.find(s => s.id === id)
|
||||||
@@ -283,11 +271,6 @@ export default {
|
|||||||
clearInterval(s.sporadicInterval)
|
clearInterval(s.sporadicInterval)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
// Reflect playback state for OS controls
|
|
||||||
if ('mediaSession' in navigator) {
|
|
||||||
try { navigator.mediaSession.playbackState = 'paused' } catch (e) { /* no-op */ }
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
startTimer () {
|
startTimer () {
|
||||||
this.timeRemaining -= 1
|
this.timeRemaining -= 1
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ import App from './App.vue'
|
|||||||
import router from './router'
|
import router from './router'
|
||||||
import vuetify from './plugins/vuetify'
|
import vuetify from './plugins/vuetify'
|
||||||
import instance from './axios'
|
import instance from './axios'
|
||||||
import './registerServiceWorker'
|
|
||||||
|
|
||||||
Vue.prototype.$http = instance
|
Vue.prototype.$http = instance
|
||||||
|
|
||||||
|
|||||||
@@ -1,45 +0,0 @@
|
|||||||
// Registers the service worker generated by @vue/cli-plugin-pwa
|
|
||||||
// This enables installability and offline caching for static assets.
|
|
||||||
|
|
||||||
if (process.env.NODE_ENV === 'production' && 'serviceWorker' in navigator) {
|
|
||||||
const register = async () => {
|
|
||||||
try {
|
|
||||||
const reg = await navigator.serviceWorker.register(`${process.env.BASE_URL}service-worker.js`)
|
|
||||||
// Listen for updates and activate immediately
|
|
||||||
reg.addEventListener('updatefound', () => {
|
|
||||||
const newWorker = reg.installing
|
|
||||||
if (!newWorker) return
|
|
||||||
newWorker.addEventListener('statechange', () => {
|
|
||||||
if (newWorker.state === 'installed' && navigator.serviceWorker.controller) {
|
|
||||||
// New content is available; refresh clients
|
|
||||||
reg.waiting && reg.waiting.postMessage({ type: 'SKIP_WAITING' })
|
|
||||||
}
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
// Claim clients after activation
|
|
||||||
navigator.serviceWorker.addEventListener('controllerchange', () => {
|
|
||||||
// optional: location.reload()
|
|
||||||
})
|
|
||||||
} catch (e) {
|
|
||||||
// eslint-disable-next-line no-console
|
|
||||||
console.warn('SW registration failed', e)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
register()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Minimal Media Session API wiring so playback can continue in background on supported browsers
|
|
||||||
export function setMediaSessionMetadata (opts = {}) {
|
|
||||||
if ('mediaSession' in navigator) {
|
|
||||||
const md = new window.MediaMetadata({
|
|
||||||
title: opts.title || 'Noisedash',
|
|
||||||
artist: opts.artist || 'Ambient Generator',
|
|
||||||
album: opts.album || 'Noisedash',
|
|
||||||
artwork: opts.artwork || [
|
|
||||||
{ src: '/favicon.ico', sizes: '64x64', type: 'image/x-icon' }
|
|
||||||
]
|
|
||||||
})
|
|
||||||
navigator.mediaSession.metadata = md
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -3,48 +3,6 @@ module.exports = {
|
|||||||
'vuetify'
|
'vuetify'
|
||||||
],
|
],
|
||||||
devServer: {
|
devServer: {
|
||||||
host: '0.0.0.0',
|
proxy: 'http://localhost:1432'
|
||||||
port: 8080,
|
|
||||||
allowedHosts: 'all',
|
|
||||||
proxy: 'http://localhost:1432'
|
|
||||||
},
|
|
||||||
pwa: {
|
|
||||||
name: 'Noisedash',
|
|
||||||
themeColor: '#121212',
|
|
||||||
backgroundColor: '#121212',
|
|
||||||
display: 'standalone',
|
|
||||||
startUrl: '/',
|
|
||||||
manifestOptions: {
|
|
||||||
short_name: 'Noisedash',
|
|
||||||
description: 'Ambient noise generator',
|
|
||||||
categories: ['music', 'audio', 'productivity'],
|
|
||||||
display_override: ['standalone', 'browser'],
|
|
||||||
orientation: 'any'
|
|
||||||
},
|
|
||||||
iconPaths: {
|
|
||||||
favicon32: 'favicon.ico',
|
|
||||||
favicon16: 'favicon.ico',
|
|
||||||
appleTouchIcon: 'favicon.ico',
|
|
||||||
maskIcon: 'favicon.ico',
|
|
||||||
msTileImage: 'favicon.ico'
|
|
||||||
},
|
|
||||||
workboxPluginMode: 'GenerateSW',
|
|
||||||
workboxOptions: {
|
|
||||||
cleanupOutdatedCaches: true,
|
|
||||||
skipWaiting: true,
|
|
||||||
clientsClaim: true,
|
|
||||||
offlineGoogleAnalytics: false,
|
|
||||||
runtimeCaching: [
|
|
||||||
{
|
|
||||||
urlPattern: /\.(?:png|jpg|jpeg|svg|gif|webp|mp3|wav|ogg)$/,
|
|
||||||
handler: 'CacheFirst',
|
|
||||||
options: {
|
|
||||||
cacheName: 'assets-cache',
|
|
||||||
expiration: { maxEntries: 60, maxAgeSeconds: 7 * 24 * 60 * 60 },
|
|
||||||
cacheableResponse: { statuses: [0, 200] }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user