Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
42 commits
Select commit Hold shift + click to select a range
1a9c129
Remove ApiClient
tatevikg1 Nov 27, 2025
21d04aa
Login
tatevikg1 Feb 25, 2026
722c663
PublicRoute Attr
tatevikg1 Feb 25, 2026
9e0378b
SessionAuthenticator
tatevikg1 Feb 25, 2026
fea446d
Vue dashboard
tatevikg1 Feb 25, 2026
718c53a
Icons
tatevikg1 Feb 25, 2026
3cd30a2
Css
tatevikg1 Feb 25, 2026
b5e79c4
Css + bootstrap
tatevikg1 Feb 25, 2026
b49da9b
Sidebar
tatevikg1 Feb 26, 2026
23a30de
grid
tatevikg1 Feb 26, 2026
e5a566d
sticky
tatevikg1 Feb 26, 2026
3113743
routes
tatevikg1 Feb 26, 2026
c97e79e
Update rest-client and core
tatevikg1 Feb 27, 2026
f698b69
Redirect to home if logged in
tatevikg1 Feb 27, 2026
3af41b4
Exclude api from firewall
tatevikg1 Feb 27, 2026
a9477f4
ApiSessionListener
tatevikg1 Mar 2, 2026
6d656e9
SubscribersController
tatevikg1 Mar 2, 2026
582cdfe
testing
tatevikg1 Mar 2, 2026
56b2e6f
Use tailwind
tatevikg1 Mar 2, 2026
183507f
Subscribers
tatevikg1 Mar 2, 2026
0bfde9d
Subscribers pass data
tatevikg1 Mar 3, 2026
04b4195
Color ext-wf1
tatevikg1 Mar 4, 2026
a781f1a
Align icon with text
tatevikg1 Mar 4, 2026
5be4574
Mobile
tatevikg1 Mar 4, 2026
aa6f146
Filter subscribers
tatevikg1 Mar 4, 2026
58cca2a
Filter subscribers
tatevikg1 Mar 9, 2026
9e22373
Fix: created
tatevikg1 Mar 9, 2026
77c007b
ref: SubscribersController
tatevikg1 Mar 9, 2026
9989080
Fix: loading for SPA
tatevikg1 Mar 9, 2026
e18b4cc
Fix: test
tatevikg1 Mar 9, 2026
0244cc1
Fix: style
tatevikg1 Mar 9, 2026
f8b6b27
Fix: tests
tatevikg1 Mar 9, 2026
781b71a
spa.html.twig
tatevikg1 Mar 9, 2026
4e40cf2
subscriberFilters
tatevikg1 Mar 9, 2026
ef19e4e
find subscriber by email
tatevikg1 Mar 9, 2026
2b22265
find subscriber by email, foreign key and unique id
tatevikg1 Mar 9, 2026
1257a70
download csv
tatevikg1 Mar 9, 2026
cdb7416
Export CSV
tatevikg1 Mar 9, 2026
5732a0c
Apache config
tatevikg1 Mar 10, 2026
f1df0b7
Export filtered
tatevikg1 Mar 10, 2026
1e18e5f
Sort
tatevikg1 Mar 10, 2026
36212bb
Fix: logout
tatevikg1 Mar 11, 2026
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
26 changes: 26 additions & 0 deletions apache/web-frontend.conf
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
<VirtualHost *:80>
ServerName frontend.phplist.local
ServerAdmin webmaster@localhost

DocumentRoot /{pathToTheProject}/web-frontend/public

ErrorLog ${APACHE_LOG_DIR}/web-frontend-error.log
CustomLog ${APACHE_LOG_DIR}/web-frontend-access.log combined

<Directory /{pathToTheProject}/web-frontend/public>
Options FollowSymLinks
AllowOverride All
Require all granted
DirectoryIndex app.php
</Directory>

<FilesMatch "^\.">
Require all denied
</FilesMatch>

<IfModule proxy_fcgi_module>
<FilesMatch \.php$>
SetHandler "proxy:unix:/run/php/php8.1-fpm.sock|fcgi://localhost"
</FilesMatch>
</IfModule>
</VirtualHost>
8 changes: 6 additions & 2 deletions assets/app.js
Original file line number Diff line number Diff line change
@@ -1,9 +1,13 @@
import './styles/app.css';
import { createApp } from 'vue';
import App from './vue/App.vue';
import { router } from './router';

// Mount the main app if the element exists
const appElement = document.getElementById('vue-app');

if (appElement) {
createApp(App).mount('#vue-app');
const app = createApp(App);
app.use(router);
app.mount(appElement);
}

Binary file added assets/images/avatar.jpg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added assets/images/logo-48.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added assets/images/logo.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
18 changes: 18 additions & 0 deletions assets/router/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
import { createRouter, createWebHistory } from 'vue-router';
import DashboardView from '../vue/views/DashboardView.vue'
import SubscribersView from '../vue/views/SubscribersView.vue'

export const router = createRouter({
history: createWebHistory(),
routes: [
{ path: '/', name: 'dashboard', component: DashboardView, meta: { title: 'Dashboard' } },
{ path: '/subscribers', name: 'subscribers', component: SubscribersView, meta: { title: 'Subscribers' } },
{ path: '/:pathMatch(.*)*', redirect: '/' },
],
});

router.afterEach((to) => {
const defaultTitle = 'phpList';
const pageTitle = to.meta.title;
document.title = pageTitle ? `${defaultTitle} - ${pageTitle}` : defaultTitle;
});
16 changes: 16 additions & 0 deletions assets/styles/app.css
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
@import "tailwindcss";
@tailwind utilities;

@theme {
--color-primary: #2563eb;
--color-secondary: #6b7280;
--color-success: #16a34a;
--color-danger: #dc2626;
--color-info: #0891b2;
--color-ext-wf1: #543ff6; /** indigo-500 **/
--color-ext-wf2: #eef2ff; /** indigo-50 **/
--color-ext-wf3: #303F9F; /** indigo-700 **/
}

@source "../../templates/**/*.twig";
@source "../**/*.vue";
29 changes: 8 additions & 21 deletions assets/vue/App.vue
Original file line number Diff line number Diff line change
@@ -1,26 +1,13 @@
<template>
<div>
<h2>Hello from Vue</h2>
<p>{{ message }}</p>
<div class="min-h-screen bg-[#F8FAFC] flex text-slate-900 font-sans">
<AppSidebar />

<div class="flex flex-col flex-1 min-h-screen">
<RouterView />
</div>
</div>
</template>

<script>
export default {
name: 'App',
data() {
return {
message: 'This is a reusable component!'
}
},
created() {
console.log('App component created');
},
mounted() {
console.log('App component mounted');
},
updated() {
console.log('App component updated');
}
}
<script setup>
import AppSidebar from './components/sidebar/AppSidebar.vue'
</script>
28 changes: 28 additions & 0 deletions assets/vue/components/base/BaseBadge.vue
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
<!-- assets/vue/components/base/BaseBadge.vue -->
<template>
<span :class="badgeClass">
<slot />
</span>
<!-- Renders a Tailwind badge; styling controlled via variant prop -->
</template>

<script setup>
import { computed } from 'vue'
const props = defineProps({
variant: {
type: String,
default: 'neutral', // neutral | counter
},
})

const badgeClass = computed(() => {
const base = 'inline-flex items-center px-2 py-0.5 rounded-full text-xs font-medium';
switch (props.variant) {
case 'counter':
return `${base} bg-indigo-50 text-ext-wf3 border border-indigo-100`;
case 'neutral':
default:
return `${base} bg-gray-100 text-gray-800`;
}
})
</script>
35 changes: 35 additions & 0 deletions assets/vue/components/base/BaseButton.vue
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
<!-- assets/vue/components/base/BaseButton.vue -->
<template>
<button
:class="buttonClass"
type="button"
v-bind="$attrs"
>
<slot />
</button>
</template>

<script setup>
import { computed } from 'vue'

const props = defineProps({
variant: {
type: String,
default: 'primary', // primary | secondary | ghost
},
})

const buttonClass = computed(() => {
const base = 'inline-flex items-center px-4 py-2 border border-transparent text-sm font-medium rounded-md shadow-sm focus:outline-none focus:ring-2 focus:ring-offset-2';
switch (props.variant) {
case 'secondary':
return `${base} text-gray-700 bg-white border-gray-300 hover:bg-gray-50 focus:ring-blue-500`;
case 'ghost':
return 'inline-flex items-center px-4 py-2 border border-transparent text-sm font-medium rounded-md text-blue-600 bg-transparent hover:bg-blue-50 focus:outline-none';
case 'primary':
default:
return `${base} text-white bg-blue-600 hover:bg-blue-700 focus:ring-blue-500`;
}
})
</script>

34 changes: 34 additions & 0 deletions assets/vue/components/base/BaseCard.vue
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
<template>
<div :class="cardClasses">
<div :class="bodyClasses">
<slot />
</div>
</div>

</template>

<script setup>
const props = defineProps({
variant: {
type: String,
default: 'default', // default | subtle | danger | success
},
})

const cardVariantMap = {
default: 'rounded-lg shadow-sm border border-gray-100 bg-white',
subtle: 'rounded-lg shadow-sm border-0 bg-gray-50',
danger: 'rounded-lg shadow-sm border-0 bg-red-600 text-white',
success: 'rounded-lg shadow-sm border-0 bg-green-600 text-white',
}

const bodyVariantMap = {
default: 'p-4',
subtle: 'p-4',
danger: 'p-4',
success: 'p-4',
}

const cardClasses = cardVariantMap[props.variant] || cardVariantMap.default
const bodyClasses = bodyVariantMap[props.variant] || bodyVariantMap.default
</script>
Loading
Loading