Compare commits

...

11 Commits

Author SHA1 Message Date
Kay Thomas
a2b6f1245c Merge pull request #18 from kaythomas0/dev
v0.4.0
2022-05-26 20:51:04 -07:00
Kay Thomas
9400959852 Fix watchable settings 2022-05-26 19:59:01 -07:00
Kay Thomas
72a7f99850 Update CONTRIBUTING.md 2022-05-26 18:59:25 -07:00
Kay Thomas
f40e6d47f4 Run npm update 2022-05-26 17:30:32 -07:00
Kay Thomas
4ab595f3a1 Fix linting errors 2022-05-26 17:08:00 -07:00
Kay Thomas
625ce328dd Update vue cli to version 5 2022-05-26 17:02:33 -07:00
Kay Thomas
411270b850 Update package-lock.json 2022-05-26 16:52:17 -07:00
Kevin Thomas
2cc5051ab4 Warn on unsaved changes 2022-05-25 21:09:41 -07:00
Kay Thomas
ff902fac16 Update FUNDING.yml 2022-05-14 19:29:28 -07:00
Kevin Thomas
8e1a8aeb05 Update CONTRIBUTING.md 2022-05-08 17:05:22 -07:00
Kay Thomas
fa59f8ebfe Update docker-image.yml
Start building multi-platform images with the same tag names
2022-05-04 09:21:36 -07:00
21 changed files with 6126 additions and 7742 deletions

View File

@@ -8,7 +8,7 @@ module.exports = {
'@vue/standard' '@vue/standard'
], ],
parserOptions: { parserOptions: {
parser: 'babel-eslint' parser: '@babel/eslint-parser'
}, },
rules: { rules: {
'no-console': process.env.NODE_ENV === 'production' ? 'warn' : 'off', 'no-console': process.env.NODE_ENV === 'production' ? 'warn' : 'off',

1
.github/FUNDING.yml vendored
View File

@@ -1,3 +1,4 @@
# These are supported funding model platforms # These are supported funding model platforms
github: kaythomas0 github: kaythomas0
custom: "https://kaythomas.dev/cryptocurrency.html"

View File

@@ -25,18 +25,10 @@ jobs:
username: ${{ secrets.DOCKERHUB_USERNAME }} username: ${{ secrets.DOCKERHUB_USERNAME }}
password: ${{ secrets.DOCKERHUB_TOKEN }} password: ${{ secrets.DOCKERHUB_TOKEN }}
- -
name: Build and push linux/amd64 name: Build and push
uses: docker/build-push-action@v2 uses: docker/build-push-action@v2
with: with:
context: . context: .
platforms: linux/amd64 platforms: linux/amd64,linux/arm/v7,linux/arm64
push: true push: true
tags: noisedash/noisedash:latest tags: noisedash/noisedash:latest,noisedash/noisedash:${{ github.ref_name }}
-
name: Build and push linux/arm/v7
uses: docker/build-push-action@v2
with:
context: .
platforms: linux/arm/v7
push: true
tags: noisedash/noisedash:latest-armv7

View File

@@ -1,4 +1,6 @@
## Project setup ## Project setup
Requires [Node](https://nodejs.org/en/download/) and [Vue CLI](https://cli.vuejs.org/guide/installation.html)
``` ```
npm install npm install
``` ```
@@ -23,5 +25,22 @@ npm run build
npm run lint npm run lint
``` ```
### Directory Summary
Here are some of the more important files and directories:
* `config/default.json`: Contains the default configuration file
* `server/*`: Where all of the node server related code is
* `server/app.js`: The main server file where server settings are set
* `server/db.js`: Where the database is created
* `server/logger.js`: Where the logger is created and configured
* `server/bin/www.js`: The entry point of the server application (what you run to start the server)
* `server/boot/*`: These are run on server startup
* `server/routes/*`: Where all of the server routes and logic are defined
* `src/*`: Contains all the frontend code
* `src/components/*`: Where all of the Vue components are defined, split into vue and js files for each component
* `src/router/index.js`: Where all the routing and route-protection logic is defined
* `src/views/*`: Contains all the views
### Customize configuration ### Customize configuration
See [Configuration Reference](https://cli.vuejs.org/config/). See [Configuration Reference](https://cli.vuejs.org/config/).

13574
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -1,18 +1,14 @@
{ {
"name": "noisedash", "name": "noisedash",
"descriptions": "Self-hostable web tool for generating ambient noises", "version": "0.4.0",
"homepage": "https://github.com/kaythomas0/noisedash",
"bugs": "https://github.com/kaythomas0/noisedash/issues",
"license": "AGPL-3.0-or-later",
"author": "Kay Thomas <kaythomas@pm.me> (https://kaythomas.dev)",
"version": "0.2.0",
"private": true, "private": true,
"author": "Kay Thomas <kaythomas@pm.me> (https://kaythomas.dev)",
"scripts": { "scripts": {
"serve": "vue-cli-service serve", "serve": "vue-cli-service serve",
"server": "node server/bin/www.js",
"server-prod": "NODE_ENV=production node server/bin/www.js",
"build": "vue-cli-service build", "build": "vue-cli-service build",
"lint": "vue-cli-service lint" "lint": "vue-cli-service lint",
"server": "node server/bin/www.js",
"server-prod": "NODE_ENV=production node server/bin/www.js"
}, },
"dependencies": { "dependencies": {
"@vscode/sqlite3": "^5.0.8", "@vscode/sqlite3": "^5.0.8",
@@ -20,9 +16,9 @@
"config": "^3.3.6", "config": "^3.3.6",
"connect-history-api-fallback": "^1.6.0", "connect-history-api-fallback": "^1.6.0",
"cookie-parser": "^1.4.5", "cookie-parser": "^1.4.5",
"core-js": "^3.19.1", "core-js": "^3.22.7",
"express": "^4.17.1", "express": "^4.18.1",
"express-session": "^1.17.2", "express-session": "^1.17.3",
"multer": "^1.4.3", "multer": "^1.4.3",
"passport": "^0.4.1", "passport": "^0.4.1",
"passport-local": "^1.0.0", "passport-local": "^1.0.0",
@@ -30,28 +26,33 @@
"session-file-store": "^1.5.0", "session-file-store": "^1.5.0",
"tone": "^14.7.77", "tone": "^14.7.77",
"vue": "^2.6.11", "vue": "^2.6.11",
"vue-router": "^3.5.3", "vue-router": "^3.5.4",
"vuetify": "^2.5.12", "vuetify": "^2.6.6",
"winston": "^3.3.3" "winston": "^3.3.3"
}, },
"devDependencies": { "devDependencies": {
"@vue/cli-plugin-babel": "^4.5.15", "@babel/core": "^7.12.16",
"@vue/cli-plugin-eslint": "^4.5.15", "@babel/eslint-parser": "^7.12.16",
"@vue/cli-plugin-router": "^4.5.15", "@vue/cli-plugin-babel": "^5.0.4",
"@vue/cli-service": "^4.5.15", "@vue/cli-plugin-eslint": "^5.0.4",
"@vue/eslint-config-standard": "^5.1.2", "@vue/cli-plugin-router": "^5.0.4",
"babel-eslint": "^10.1.0", "@vue/cli-service": "^5.0.4",
"eslint": "^6.7.2", "@vue/eslint-config-standard": "^6.1.0",
"eslint": "^7.32.0",
"eslint-plugin-html": "^6.2.0", "eslint-plugin-html": "^6.2.0",
"eslint-plugin-import": "^2.25.2", "eslint-plugin-import": "^2.25.3",
"eslint-plugin-node": "^11.1.0", "eslint-plugin-node": "^11.1.0",
"eslint-plugin-promise": "^4.2.1", "eslint-plugin-promise": "^5.1.0",
"eslint-plugin-standard": "^4.0.0", "eslint-plugin-standard": "^4.0.0",
"eslint-plugin-vue": "^6.2.2", "eslint-plugin-vue": "^8.0.3",
"sass": "~1.32.0", "sass": "~1.32.0",
"sass-loader": "^10.0.0", "sass-loader": "^10.0.0",
"vue-cli-plugin-vuetify": "^2.4.3", "vue-cli-plugin-vuetify": "^2.5.0",
"vue-template-compiler": "^2.6.11", "vue-template-compiler": "^2.6.11",
"vuetify-loader": "^1.7.3" "vuetify-loader": "^1.7.3"
} },
"bugs": "https://github.com/kaythomas0/noisedash/issues",
"descriptions": "Self-hostable web tool for generating ambient noises",
"homepage": "https://github.com/kaythomas0/noisedash",
"license": "AGPL-3.0-or-later"
} }

View File

@@ -24,7 +24,7 @@
v-model="changePasswordDialog" v-model="changePasswordDialog"
max-width="600px" max-width="600px"
> >
<template v-slot:activator="{ on, attrs }"> <template #activator="{ on, attrs }">
<v-btn <v-btn
class="my-3" class="my-3"
v-bind="attrs" v-bind="attrs"
@@ -89,7 +89,7 @@
> >
{{ snackbarText }} {{ snackbarText }}
<template v-slot:action="{ attrs }"> <template #action="{ attrs }">
<v-btn <v-btn
text text
v-bind="attrs" v-bind="attrs"

View File

@@ -12,7 +12,7 @@
v-model="registerUserDialog" v-model="registerUserDialog"
max-width="600px" max-width="600px"
> >
<template v-slot:activator="{ on, attrs }"> <template #activator="{ on, attrs }">
<v-btn <v-btn
v-bind="attrs" v-bind="attrs"
v-on="on" v-on="on"
@@ -139,7 +139,7 @@
> >
{{ snackbarText }} {{ snackbarText }}
<template v-slot:action="{ attrs }"> <template #action="{ attrs }">
<v-btn <v-btn
text text
v-bind="attrs" v-bind="attrs"

View File

@@ -63,7 +63,7 @@
</v-list-item> </v-list-item>
</v-list-item-group> </v-list-item-group>
</v-list> </v-list>
<template v-slot:append> <template #append>
<v-btn <v-btn
block block
href="https://github.com/kaythomas0/noisedash" href="https://github.com/kaythomas0/noisedash"

View File

@@ -50,7 +50,7 @@
> >
{{ snackbarText }} {{ snackbarText }}
<template v-slot:action="{ attrs }"> <template #action="{ attrs }">
<v-btn <v-btn
text text
v-bind="attrs" v-bind="attrs"

View File

@@ -66,14 +66,14 @@
class="mx-3 my-3" class="mx-3 my-3"
@click="updateProfile" @click="updateProfile"
> >
Save Profile {{ unsavedWork ? 'Save Profile*' : 'Save Profile' }}
</v-btn> </v-btn>
<v-dialog <v-dialog
v-model="profileDialog" v-model="profileDialog"
max-width="600px" max-width="600px"
> >
<template v-slot:activator="{ on, attrs }"> <template #activator="{ on, attrs }">
<v-btn <v-btn
v-bind="attrs" v-bind="attrs"
class="mx-3 my-3" class="mx-3 my-3"
@@ -127,7 +127,7 @@
v-model="profileMoreDialog" v-model="profileMoreDialog"
max-width="600px" max-width="600px"
> >
<template v-slot:activator="{ on, attrs }"> <template #activator="{ on, attrs }">
<v-btn <v-btn
v-bind="attrs" v-bind="attrs"
class="mx-3 my-3" class="mx-3 my-3"
@@ -339,7 +339,7 @@
<v-dialog <v-dialog
v-model="recordingDialog" v-model="recordingDialog"
max-width="600px" max-width="600px"
persistent="true" persistent
> >
<v-card> <v-card>
<v-card-title> <v-card-title>
@@ -365,6 +365,36 @@
</v-card-actions> </v-card-actions>
</v-card> </v-card>
</v-dialog> </v-dialog>
<v-dialog
v-model="confirmSwitchProfileDialog"
max-width="600px"
persistent
>
<v-card>
<v-card-title>
<span class="text-h5">Save Profile?</span>
</v-card-title>
<v-card-text>
You have unsaved work on your current profile. Would you like to save it before switching?
</v-card-text>
<v-card-actions>
<v-spacer />
<v-btn
text
@click="discardChanges"
>
Discard Changes
</v-btn>
<v-btn
text
@click="saveChanges"
>
Save Profile
</v-btn>
</v-card-actions>
</v-card>
</v-dialog>
</v-col> </v-col>
<v-col cols="12"> <v-col cols="12">
@@ -656,7 +686,7 @@
v-model="addSampleDialog" v-model="addSampleDialog"
max-width="600px" max-width="600px"
> >
<template v-slot:activator="{ on, attrs }"> <template #activator="{ on, attrs }">
<v-btn <v-btn
v-bind="attrs" v-bind="attrs"
class="mx-3 my-3 mb-5" class="mx-3 my-3 mb-5"
@@ -716,7 +746,7 @@
v-model="uploadSampleDialog" v-model="uploadSampleDialog"
max-width="600px" max-width="600px"
> >
<template v-slot:activator="{ on, attrs }"> <template #activator="{ on, attrs }">
<v-btn <v-btn
v-bind="attrs" v-bind="attrs"
class="mx-3 my-3 mb-5" class="mx-3 my-3 mb-5"
@@ -780,7 +810,7 @@
v-model="editSampleDialog" v-model="editSampleDialog"
max-width="600px" max-width="600px"
> >
<template v-slot:activator="{ on, attrs }"> <template #activator="{ on, attrs }">
<v-btn <v-btn
v-bind="attrs" v-bind="attrs"
class="mx-3 my-3 mb-5" class="mx-3 my-3 mb-5"
@@ -896,7 +926,7 @@
> >
{{ infoSnackbarText }} {{ infoSnackbarText }}
<template v-slot:action="{ attrs }"> <template #action="{ attrs }">
<v-btn <v-btn
text text
v-bind="attrs" v-bind="attrs"
@@ -914,7 +944,7 @@
> >
{{ errorSnackbarText }} {{ errorSnackbarText }}
<template v-slot:action="{ attrs }"> <template #action="{ attrs }">
<v-btn <v-btn
text text
v-bind="attrs" v-bind="attrs"

View File

@@ -1,5 +1,5 @@
export default { export default {
name: 'Admin', name: 'Account',
data: () => ({ data: () => ({
currentUser: {}, currentUser: {},

View File

@@ -1,4 +1,6 @@
export default { export default {
name: 'Login',
data: () => ({ data: () => ({
valid: false, valid: false,
username: '', username: '',

View File

@@ -70,6 +70,11 @@ export default {
recordedProfile: {}, recordedProfile: {},
recordingFileName: '', recordingFileName: '',
isRecordingValid: false, isRecordingValid: false,
unsavedWork: false,
saveProfileText: 'Save Profile',
unwatch: null,
confirmSwitchProfileDialog: false,
activeProfile: {},
errorSnackbar: false, errorSnackbar: false,
errorSnackbarText: '', errorSnackbarText: '',
rules: { rules: {
@@ -94,6 +99,33 @@ export default {
} }
}) })
return samples return samples
},
changeableSettings: function () {
const settings = [
this.isTimerEnabled,
this.hours,
this.minutes,
this.seconds,
this.volume,
this.noiseColor,
this.isFilterEnabled,
this.filterType,
this.filterCutoff,
this.isLFOFilterCutoffEnabled,
this.lfoFilterCutoffFrequency,
this.lfoFilterCutoffRange,
this.isTremoloEnabled,
this.tremoloDepth,
this.tremoloFrequency,
this.isTimerEnabled,
this.loadedSamples
]
this.loadedSamples.forEach(s => {
settings.push(s.volume)
})
return settings
} }
}, },
created () { created () {
@@ -294,6 +326,7 @@ export default {
if (response.status === 200) { if (response.status === 200) {
this.profileDialog = false this.profileDialog = false
this.populateProfileItems(response.data.id) this.populateProfileItems(response.data.id)
this.unsavedWork = false
this.infoSnackbarText = 'Profile Saved' this.infoSnackbarText = 'Profile Saved'
this.infoSnackbar = true this.infoSnackbar = true
} }
@@ -322,6 +355,7 @@ export default {
samples: this.loadedSamples samples: this.loadedSamples
}).then(response => { }).then(response => {
if (response.status === 200) { if (response.status === 200) {
this.unsavedWork = false
this.infoSnackbarText = 'Profile Saved' this.infoSnackbarText = 'Profile Saved'
this.infoSnackbar = true this.infoSnackbar = true
} }
@@ -332,33 +366,47 @@ export default {
}) })
}, },
loadProfile () { loadProfile () {
this.$http.get('/profiles/'.concat(this.selectedProfile.id)) if (this.unsavedWork) {
.then(response => { this.confirmSwitchProfileDialog = true
if (response.status === 200) { } else {
const profile = response.data.profile this.$http.get('/profiles/'.concat(this.selectedProfile.id))
.then(response => {
if (response.status === 200) {
const profile = response.data.profile
this.isTimerEnabled = profile.isTimerEnabled this.isTimerEnabled = profile.isTimerEnabled
this.duration = profile.duration this.duration = profile.duration
this.volume = profile.volume this.volume = profile.volume
this.noiseColor = profile.noiseColor this.noiseColor = profile.noiseColor
this.isFilterEnabled = profile.isFilterEnabled this.isFilterEnabled = profile.isFilterEnabled
this.filterType = profile.filterType this.filterType = profile.filterType
this.filterCutoff = profile.filterCutoff this.filterCutoff = profile.filterCutoff
this.isLFOFilterCutoffEnabled = profile.isLFOFilterCutoffEnabled this.isLFOFilterCutoffEnabled = profile.isLFOFilterCutoffEnabled
this.lfoFilterCutoffFrequency = profile.lfoFilterCutoffFrequency this.lfoFilterCutoffFrequency = profile.lfoFilterCutoffFrequency
this.lfoFilterCutoffRange[0] = profile.lfoFilterCutoffLow this.lfoFilterCutoffRange[0] = profile.lfoFilterCutoffLow
this.lfoFilterCutoffRange[1] = profile.lfoFilterCutoffHigh this.lfoFilterCutoffRange[1] = profile.lfoFilterCutoffHigh
this.isTremoloEnabled = profile.isTremoloEnabled this.isTremoloEnabled = profile.isTremoloEnabled
this.tremoloFrequency = profile.tremoloFrequency this.tremoloFrequency = profile.tremoloFrequency
this.tremoloDepth = profile.tremoloDepth this.tremoloDepth = profile.tremoloDepth
this.loadedSamples = profile.samples this.loadedSamples = profile.samples
}
}) this.activeProfile = profile
.catch(() => {
this.errorSnackbarText = 'Error Loading Profile' if (this.unwatch) {
this.errorSnackbar = true this.unwatch()
}) }
this.unwatch = this.$watch('changeableSettings', function () {
this.unsavedWork = true
})
}
})
.catch(() => {
this.errorSnackbarText = 'Error Loading Profile'
this.errorSnackbar = true
})
}
}, },
deleteProfile () { deleteProfile () {
this.$http.delete('/profiles/'.concat(this.selectedProfile.id)) this.$http.delete('/profiles/'.concat(this.selectedProfile.id))
@@ -749,6 +797,17 @@ export default {
clearInterval(this.recordingInterval) clearInterval(this.recordingInterval)
this.recordingDialog = false this.recordingDialog = false
this.stop() this.stop()
},
discardChanges () {
this.unsavedWork = false
this.loadProfile()
this.confirmSwitchProfileDialog = false
},
saveChanges () {
// Set active profile back to previously selected one before saving
this.selectedProfile = this.profileItems.find(p => p.text === this.activeProfile.name)
this.updateProfile()
this.confirmSwitchProfileDialog = false
} }
} }
} }

View File

@@ -1,4 +1,6 @@
export default { export default {
name: 'Register',
data: () => ({ data: () => ({
valid: false, valid: false,
name: '', name: '',

View File

@@ -1,15 +1,15 @@
<template> <template>
<Account /> <AccountPage />
</template> </template>
<script> <script>
import Account from '../components/Account' import AccountPage from '../components/AccountPage'
export default { export default {
name: 'AccountView', name: 'AccountView',
components: { components: {
Account AccountPage
} }
} }
</script> </script>

View File

@@ -1,15 +1,15 @@
<template> <template>
<Admin /> <AdminPage />
</template> </template>
<script> <script>
import Admin from '../components/Admin' import AdminPage from '../components/AdminPage'
export default { export default {
name: 'AdminView', name: 'AdminView',
components: { components: {
Admin AdminPage
} }
} }
</script> </script>

View File

@@ -1,15 +1,15 @@
<template> <template>
<Noise /> <NoisePage />
</template> </template>
<script> <script>
import Noise from '../components/Noise' import NoisePage from '../components/NoisePage'
export default { export default {
name: 'HomeView', name: 'HomeView',
components: { components: {
Noise NoisePage
} }
} }
</script> </script>

View File

@@ -1,15 +1,15 @@
<template> <template>
<Login /> <LoginPage />
</template> </template>
<script> <script>
import Login from '../components/Login' import LoginPage from '../components/LoginPage'
export default { export default {
name: 'LoginView', name: 'LoginView',
components: { components: {
Login LoginPage
} }
} }
</script> </script>

View File

@@ -1,15 +1,15 @@
<template> <template>
<Register /> <RegisterPage />
</template> </template>
<script> <script>
import Register from '../components/Register' import RegisterPage from '../components/RegisterPage'
export default { export default {
name: 'RegisterView', name: 'RegisterView',
components: { components: {
Register RegisterPage
} }
} }
</script> </script>