Added Teams
Minor Rework of Plantafel
This commit is contained in:
31
backend/db/migrations/0028_teams.sql
Normal file
31
backend/db/migrations/0028_teams.sql
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
CREATE TABLE "teams" (
|
||||||
|
"id" bigint PRIMARY KEY GENERATED BY DEFAULT AS IDENTITY (sequence name "teams_id_seq" INCREMENT BY 1 MINVALUE 1 MAXVALUE 9223372036854775807 START WITH 1 CACHE 1),
|
||||||
|
"created_at" timestamp with time zone DEFAULT now() NOT NULL,
|
||||||
|
"tenant" bigint NOT NULL,
|
||||||
|
"name" text NOT NULL,
|
||||||
|
"description" text,
|
||||||
|
"branch" bigint,
|
||||||
|
"archived" boolean DEFAULT false NOT NULL,
|
||||||
|
"updated_at" timestamp with time zone,
|
||||||
|
"updated_by" uuid
|
||||||
|
);
|
||||||
|
--> statement-breakpoint
|
||||||
|
CREATE TABLE "auth_profile_teams" (
|
||||||
|
"created_at" timestamp with time zone DEFAULT now() NOT NULL,
|
||||||
|
"profile_id" uuid NOT NULL,
|
||||||
|
"team_id" bigint NOT NULL,
|
||||||
|
"created_by" uuid,
|
||||||
|
CONSTRAINT "auth_profile_teams_profile_id_team_id_pk" PRIMARY KEY("profile_id","team_id")
|
||||||
|
);
|
||||||
|
--> statement-breakpoint
|
||||||
|
ALTER TABLE "teams" ADD CONSTRAINT "teams_tenant_tenants_id_fk" FOREIGN KEY ("tenant") REFERENCES "public"."tenants"("id") ON DELETE no action ON UPDATE no action;
|
||||||
|
--> statement-breakpoint
|
||||||
|
ALTER TABLE "teams" ADD CONSTRAINT "teams_branch_branches_id_fk" FOREIGN KEY ("branch") REFERENCES "public"."branches"("id") ON DELETE no action ON UPDATE no action;
|
||||||
|
--> statement-breakpoint
|
||||||
|
ALTER TABLE "teams" ADD CONSTRAINT "teams_updated_by_auth_users_id_fk" FOREIGN KEY ("updated_by") REFERENCES "public"."auth_users"("id") ON DELETE no action ON UPDATE no action;
|
||||||
|
--> statement-breakpoint
|
||||||
|
ALTER TABLE "auth_profile_teams" ADD CONSTRAINT "auth_profile_teams_profile_id_auth_profiles_id_fk" FOREIGN KEY ("profile_id") REFERENCES "public"."auth_profiles"("id") ON DELETE cascade ON UPDATE no action;
|
||||||
|
--> statement-breakpoint
|
||||||
|
ALTER TABLE "auth_profile_teams" ADD CONSTRAINT "auth_profile_teams_team_id_teams_id_fk" FOREIGN KEY ("team_id") REFERENCES "public"."teams"("id") ON DELETE cascade ON UPDATE no action;
|
||||||
|
--> statement-breakpoint
|
||||||
|
ALTER TABLE "auth_profile_teams" ADD CONSTRAINT "auth_profile_teams_created_by_auth_users_id_fk" FOREIGN KEY ("created_by") REFERENCES "public"."auth_users"("id") ON DELETE no action ON UPDATE no action;
|
||||||
1
backend/db/migrations/0029_events_quick.sql
Normal file
1
backend/db/migrations/0029_events_quick.sql
Normal file
@@ -0,0 +1 @@
|
|||||||
|
ALTER TABLE "events" ADD COLUMN "quick" boolean DEFAULT false NOT NULL;
|
||||||
@@ -197,6 +197,20 @@
|
|||||||
"when": 1774602000000,
|
"when": 1774602000000,
|
||||||
"tag": "0027_product_supplier_link",
|
"tag": "0027_product_supplier_link",
|
||||||
"breakpoints": true
|
"breakpoints": true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"idx": 28,
|
||||||
|
"version": "7",
|
||||||
|
"when": 1776124800000,
|
||||||
|
"tag": "0028_teams",
|
||||||
|
"breakpoints": true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"idx": 29,
|
||||||
|
"version": "7",
|
||||||
|
"when": 1776211200000,
|
||||||
|
"tag": "0029_events_quick",
|
||||||
|
"breakpoints": true
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
30
backend/db/schema/auth_profile_teams.ts
Normal file
30
backend/db/schema/auth_profile_teams.ts
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
import { pgTable, uuid, bigint, timestamp } from "drizzle-orm/pg-core"
|
||||||
|
|
||||||
|
import { authProfiles } from "./auth_profiles"
|
||||||
|
import { teams } from "./teams"
|
||||||
|
import { authUsers } from "./auth_users"
|
||||||
|
|
||||||
|
export const authProfileTeams = pgTable(
|
||||||
|
"auth_profile_teams",
|
||||||
|
{
|
||||||
|
created_at: timestamp("created_at", { withTimezone: true })
|
||||||
|
.notNull()
|
||||||
|
.defaultNow(),
|
||||||
|
|
||||||
|
profile_id: uuid("profile_id")
|
||||||
|
.notNull()
|
||||||
|
.references(() => authProfiles.id, { onDelete: "cascade" }),
|
||||||
|
|
||||||
|
team_id: bigint("team_id", { mode: "number" })
|
||||||
|
.notNull()
|
||||||
|
.references(() => teams.id, { onDelete: "cascade" }),
|
||||||
|
|
||||||
|
created_by: uuid("created_by").references(() => authUsers.id),
|
||||||
|
},
|
||||||
|
(table) => ({
|
||||||
|
primaryKey: [table.profile_id, table.team_id],
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
|
export type AuthProfileTeam = typeof authProfileTeams.$inferSelect
|
||||||
|
export type NewAuthProfileTeam = typeof authProfileTeams.$inferInsert
|
||||||
@@ -31,6 +31,7 @@ export const events = pgTable(
|
|||||||
endDate: timestamp("endDate", { withTimezone: true }),
|
endDate: timestamp("endDate", { withTimezone: true }),
|
||||||
|
|
||||||
eventtype: text("eventtype").default("Umsetzung"),
|
eventtype: text("eventtype").default("Umsetzung"),
|
||||||
|
quick: boolean("quick").notNull().default(false),
|
||||||
|
|
||||||
project: bigint("project", { mode: "number" }), // FK follows when projects.ts exists
|
project: bigint("project", { mode: "number" }), // FK follows when projects.ts exists
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
export * from "./accounts"
|
export * from "./accounts"
|
||||||
export * from "./auth_profiles"
|
export * from "./auth_profiles"
|
||||||
export * from "./auth_profile_branches"
|
export * from "./auth_profile_branches"
|
||||||
|
export * from "./auth_profile_teams"
|
||||||
export * from "./auth_role_permisssions"
|
export * from "./auth_role_permisssions"
|
||||||
export * from "./auth_roles"
|
export * from "./auth_roles"
|
||||||
export * from "./auth_tenant_users"
|
export * from "./auth_tenant_users"
|
||||||
@@ -69,6 +70,7 @@ export * from "./staff_time_entry_connects"
|
|||||||
export * from "./staff_zeitstromtimestamps"
|
export * from "./staff_zeitstromtimestamps"
|
||||||
export * from "./statementallocations"
|
export * from "./statementallocations"
|
||||||
export * from "./tasks"
|
export * from "./tasks"
|
||||||
|
export * from "./teams"
|
||||||
export * from "./taxtypes"
|
export * from "./taxtypes"
|
||||||
export * from "./tenants"
|
export * from "./tenants"
|
||||||
export * from "./texttemplates"
|
export * from "./texttemplates"
|
||||||
|
|||||||
40
backend/db/schema/teams.ts
Normal file
40
backend/db/schema/teams.ts
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
import {
|
||||||
|
pgTable,
|
||||||
|
bigint,
|
||||||
|
timestamp,
|
||||||
|
text,
|
||||||
|
boolean,
|
||||||
|
uuid,
|
||||||
|
} from "drizzle-orm/pg-core"
|
||||||
|
|
||||||
|
import { tenants } from "./tenants"
|
||||||
|
import { authUsers } from "./auth_users"
|
||||||
|
import { branches } from "./branches"
|
||||||
|
|
||||||
|
export const teams = pgTable("teams", {
|
||||||
|
id: bigint("id", { mode: "number" })
|
||||||
|
.primaryKey()
|
||||||
|
.generatedByDefaultAsIdentity(),
|
||||||
|
|
||||||
|
createdAt: timestamp("created_at", { withTimezone: true })
|
||||||
|
.notNull()
|
||||||
|
.defaultNow(),
|
||||||
|
|
||||||
|
tenant: bigint("tenant", { mode: "number" })
|
||||||
|
.notNull()
|
||||||
|
.references(() => tenants.id),
|
||||||
|
|
||||||
|
name: text("name").notNull(),
|
||||||
|
description: text("description"),
|
||||||
|
|
||||||
|
branch: bigint("branch", { mode: "number" })
|
||||||
|
.references(() => branches.id),
|
||||||
|
|
||||||
|
archived: boolean("archived").notNull().default(false),
|
||||||
|
|
||||||
|
updatedAt: timestamp("updated_at", { withTimezone: true }),
|
||||||
|
updatedBy: uuid("updated_by").references(() => authUsers.id),
|
||||||
|
})
|
||||||
|
|
||||||
|
export type Team = typeof teams.$inferSelect
|
||||||
|
export type NewTeam = typeof teams.$inferInsert
|
||||||
@@ -93,6 +93,7 @@ export const tenants = pgTable(
|
|||||||
incomingInvoices: true,
|
incomingInvoices: true,
|
||||||
costcentres: true,
|
costcentres: true,
|
||||||
branches: true,
|
branches: true,
|
||||||
|
teams: true,
|
||||||
accounts: true,
|
accounts: true,
|
||||||
ownaccounts: true,
|
ownaccounts: true,
|
||||||
banking: true,
|
banking: true,
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import {
|
|||||||
|
|
||||||
import {and, eq, inArray} from "drizzle-orm"
|
import {and, eq, inArray} from "drizzle-orm"
|
||||||
import { enrichProfilesWithBranches } from "../../utils/profileBranches"
|
import { enrichProfilesWithBranches } from "../../utils/profileBranches"
|
||||||
|
import { enrichProfilesWithTeams } from "../../utils/profileTeams"
|
||||||
|
|
||||||
|
|
||||||
export default async function tenantRoutesInternal(server: FastifyInstance) {
|
export default async function tenantRoutesInternal(server: FastifyInstance) {
|
||||||
@@ -62,7 +63,8 @@ export default async function tenantRoutesInternal(server: FastifyInstance) {
|
|||||||
eq(authProfiles.tenant_id, tenantId),
|
eq(authProfiles.tenant_id, tenantId),
|
||||||
inArray(authProfiles.user_id, userIds)
|
inArray(authProfiles.user_id, userIds)
|
||||||
))
|
))
|
||||||
const profiles = await enrichProfilesWithBranches(server, profileRows)
|
const profilesWithBranches = await enrichProfilesWithBranches(server, profileRows)
|
||||||
|
const profiles = await enrichProfilesWithTeams(server, profilesWithBranches)
|
||||||
|
|
||||||
const combined = users.map(u => {
|
const combined = users.map(u => {
|
||||||
const profile = profiles.find(p => p.user_id === u.id)
|
const profile = profiles.find(p => p.user_id === u.id)
|
||||||
@@ -98,7 +100,8 @@ export default async function tenantRoutesInternal(server: FastifyInstance) {
|
|||||||
.from(authProfiles)
|
.from(authProfiles)
|
||||||
.where(eq(authProfiles.tenant_id, tenantId))
|
.where(eq(authProfiles.tenant_id, tenantId))
|
||||||
|
|
||||||
return await enrichProfilesWithBranches(server, profileRows)
|
const profilesWithBranches = await enrichProfilesWithBranches(server, profileRows)
|
||||||
|
return await enrichProfilesWithTeams(server, profilesWithBranches)
|
||||||
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error("/tenant/profiles ERROR:", err)
|
console.error("/tenant/profiles ERROR:", err)
|
||||||
|
|||||||
@@ -9,6 +9,11 @@ import {
|
|||||||
resolveTenantBranchIds,
|
resolveTenantBranchIds,
|
||||||
syncProfileBranches,
|
syncProfileBranches,
|
||||||
} from "../utils/profileBranches";
|
} from "../utils/profileBranches";
|
||||||
|
import {
|
||||||
|
enrichProfilesWithTeams,
|
||||||
|
resolveTenantTeamIds,
|
||||||
|
syncProfileTeams,
|
||||||
|
} from "../utils/profileTeams";
|
||||||
|
|
||||||
export default async function authProfilesRoutes(server: FastifyInstance) {
|
export default async function authProfilesRoutes(server: FastifyInstance) {
|
||||||
|
|
||||||
@@ -24,7 +29,10 @@ export default async function authProfilesRoutes(server: FastifyInstance) {
|
|||||||
return reply.code(400).send({ error: "No tenant selected" });
|
return reply.code(400).send({ error: "No tenant selected" });
|
||||||
}
|
}
|
||||||
|
|
||||||
const profile = await loadProfileWithBranches(server, id, tenantId)
|
const profileWithBranches = await loadProfileWithBranches(server, id, tenantId)
|
||||||
|
const [profile] = profileWithBranches
|
||||||
|
? await enrichProfilesWithTeams(server, [profileWithBranches])
|
||||||
|
: [null]
|
||||||
|
|
||||||
if (!profile) {
|
if (!profile) {
|
||||||
return reply.code(404).send({ error: "User not found or not in tenant" });
|
return reply.code(404).send({ error: "User not found or not in tenant" });
|
||||||
@@ -45,7 +53,7 @@ export default async function authProfilesRoutes(server: FastifyInstance) {
|
|||||||
const forbidden = [
|
const forbidden = [
|
||||||
"id", "user_id", "tenant_id", "created_at", "updated_at",
|
"id", "user_id", "tenant_id", "created_at", "updated_at",
|
||||||
"updatedAt", "updatedBy", "old_profile_id", "full_name",
|
"updatedAt", "updatedBy", "old_profile_id", "full_name",
|
||||||
"branch", "branches", "branch_ids"
|
"branch"
|
||||||
]
|
]
|
||||||
forbidden.forEach(f => delete cleaned[f])
|
forbidden.forEach(f => delete cleaned[f])
|
||||||
|
|
||||||
@@ -95,6 +103,19 @@ export default async function authProfilesRoutes(server: FastifyInstance) {
|
|||||||
],
|
],
|
||||||
body.branch_id ?? body.branch?.id ?? null
|
body.branch_id ?? body.branch?.id ?? null
|
||||||
)
|
)
|
||||||
|
const teamIds = await resolveTenantTeamIds(
|
||||||
|
server,
|
||||||
|
tenantId,
|
||||||
|
[
|
||||||
|
...(Array.isArray(body.team_ids) ? body.team_ids : []),
|
||||||
|
...(Array.isArray(body.teams) ? body.teams : []),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
delete body.branch_ids
|
||||||
|
delete body.branches
|
||||||
|
delete body.team_ids
|
||||||
|
delete body.teams
|
||||||
|
|
||||||
const updateData = {
|
const updateData = {
|
||||||
...body,
|
...body,
|
||||||
@@ -119,8 +140,12 @@ export default async function authProfilesRoutes(server: FastifyInstance) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
await syncProfileBranches(server, id, branchIds, userId)
|
await syncProfileBranches(server, id, branchIds, userId)
|
||||||
|
await syncProfileTeams(server, id, teamIds, userId)
|
||||||
|
|
||||||
const profile = await loadProfileWithBranches(server, id, tenantId)
|
const profileWithBranches = await loadProfileWithBranches(server, id, tenantId)
|
||||||
|
const [profile] = profileWithBranches
|
||||||
|
? await enrichProfilesWithTeams(server, [profileWithBranches])
|
||||||
|
: [null]
|
||||||
return profile || updated[0]
|
return profile || updated[0]
|
||||||
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
@@ -128,6 +153,9 @@ export default async function authProfilesRoutes(server: FastifyInstance) {
|
|||||||
if (err instanceof Error && ["INVALID_BRANCH_SELECTION", "INVALID_PRIMARY_BRANCH"].includes(err.message)) {
|
if (err instanceof Error && ["INVALID_BRANCH_SELECTION", "INVALID_PRIMARY_BRANCH"].includes(err.message)) {
|
||||||
return reply.code(400).send({ error: "Ungültige Niederlassungsauswahl" })
|
return reply.code(400).send({ error: "Ungültige Niederlassungsauswahl" })
|
||||||
}
|
}
|
||||||
|
if (err instanceof Error && err.message === "INVALID_TEAM_SELECTION") {
|
||||||
|
return reply.code(400).send({ error: "Ungültige Teamauswahl" })
|
||||||
|
}
|
||||||
return reply.code(500).send({ error: "Internal Server Error" })
|
return reply.code(500).send({ error: "Internal Server Error" })
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import {
|
|||||||
|
|
||||||
import {and, desc, eq, inArray} from "drizzle-orm"
|
import {and, desc, eq, inArray} from "drizzle-orm"
|
||||||
import { enrichProfilesWithBranches } from "../utils/profileBranches"
|
import { enrichProfilesWithBranches } from "../utils/profileBranches"
|
||||||
|
import { enrichProfilesWithTeams } from "../utils/profileTeams"
|
||||||
|
|
||||||
|
|
||||||
export default async function tenantRoutes(server: FastifyInstance) {
|
export default async function tenantRoutes(server: FastifyInstance) {
|
||||||
@@ -132,7 +133,8 @@ export default async function tenantRoutes(server: FastifyInstance) {
|
|||||||
eq(authProfiles.tenant_id, tenantId),
|
eq(authProfiles.tenant_id, tenantId),
|
||||||
inArray(authProfiles.user_id, userIds)
|
inArray(authProfiles.user_id, userIds)
|
||||||
))
|
))
|
||||||
const profiles = await enrichProfilesWithBranches(server, profileRows)
|
const profilesWithBranches = await enrichProfilesWithBranches(server, profileRows)
|
||||||
|
const profiles = await enrichProfilesWithTeams(server, profilesWithBranches)
|
||||||
|
|
||||||
const combined = users.map(u => {
|
const combined = users.map(u => {
|
||||||
const profile = profiles.find(p => p.user_id === u.id)
|
const profile = profiles.find(p => p.user_id === u.id)
|
||||||
@@ -167,7 +169,8 @@ export default async function tenantRoutes(server: FastifyInstance) {
|
|||||||
.from(authProfiles)
|
.from(authProfiles)
|
||||||
.where(eq(authProfiles.tenant_id, tenantId))
|
.where(eq(authProfiles.tenant_id, tenantId))
|
||||||
|
|
||||||
const data = await enrichProfilesWithBranches(server, profileRows)
|
const profilesWithBranches = await enrichProfilesWithBranches(server, profileRows)
|
||||||
|
const data = await enrichProfilesWithTeams(server, profilesWithBranches)
|
||||||
return { data }
|
return { data }
|
||||||
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ const HISTORY_ENTITY_LABELS: Record<string, string> = {
|
|||||||
incominginvoices: "Eingangsrechnungen",
|
incominginvoices: "Eingangsrechnungen",
|
||||||
files: "Dateien",
|
files: "Dateien",
|
||||||
memberrelations: "Mitgliedsverhältnisse",
|
memberrelations: "Mitgliedsverhältnisse",
|
||||||
|
teams: "Teams",
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getHistoryEntityLabel(entity: string) {
|
export function getHistoryEntityLabel(entity: string) {
|
||||||
|
|||||||
117
backend/src/utils/profileTeams.ts
Normal file
117
backend/src/utils/profileTeams.ts
Normal file
@@ -0,0 +1,117 @@
|
|||||||
|
import { and, eq, inArray } from "drizzle-orm"
|
||||||
|
import { FastifyInstance } from "fastify"
|
||||||
|
|
||||||
|
import { authProfileTeams, branches, teams } from "../../db/schema"
|
||||||
|
|
||||||
|
function normalizeTeamIds(values: any[]): number[] {
|
||||||
|
return [...new Set(
|
||||||
|
values
|
||||||
|
.map((value) => {
|
||||||
|
if (typeof value === "number") return value
|
||||||
|
if (typeof value === "string" && value.trim()) return Number(value)
|
||||||
|
if (value && typeof value === "object" && "id" in value) return Number(value.id)
|
||||||
|
return NaN
|
||||||
|
})
|
||||||
|
.filter((value) => Number.isFinite(value))
|
||||||
|
)]
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function enrichProfilesWithTeams(server: FastifyInstance, profiles: any[]) {
|
||||||
|
if (!profiles.length) return profiles
|
||||||
|
|
||||||
|
const profileIds = profiles.map((profile) => profile.id).filter(Boolean)
|
||||||
|
if (!profileIds.length) return profiles
|
||||||
|
|
||||||
|
const profileTeamRows = await server.db
|
||||||
|
.select()
|
||||||
|
.from(authProfileTeams)
|
||||||
|
.where(inArray(authProfileTeams.profile_id, profileIds))
|
||||||
|
|
||||||
|
const teamIds = [...new Set(profileTeamRows.map((row) => row.team_id).filter(Boolean))]
|
||||||
|
const teamRows = teamIds.length
|
||||||
|
? await server.db.select().from(teams).where(inArray(teams.id, teamIds))
|
||||||
|
: []
|
||||||
|
|
||||||
|
const branchIds = [...new Set(teamRows.map((team) => team.branch).filter(Boolean))]
|
||||||
|
const branchRows = branchIds.length
|
||||||
|
? await server.db.select().from(branches).where(inArray(branches.id, branchIds))
|
||||||
|
: []
|
||||||
|
|
||||||
|
const branchMap = new Map(branchRows.map((branch) => [branch.id, branch]))
|
||||||
|
const teamMap = new Map(teamRows.map((team) => [
|
||||||
|
team.id,
|
||||||
|
{
|
||||||
|
...team,
|
||||||
|
branch: team.branch ? branchMap.get(team.branch) || null : null,
|
||||||
|
},
|
||||||
|
]))
|
||||||
|
const teamIdsByProfile = new Map<string, number[]>()
|
||||||
|
|
||||||
|
for (const row of profileTeamRows) {
|
||||||
|
const current = teamIdsByProfile.get(row.profile_id) || []
|
||||||
|
current.push(row.team_id)
|
||||||
|
teamIdsByProfile.set(row.profile_id, current)
|
||||||
|
}
|
||||||
|
|
||||||
|
return profiles.map((profile) => {
|
||||||
|
const assignedTeamIds = [...new Set(teamIdsByProfile.get(profile.id) || [])]
|
||||||
|
return {
|
||||||
|
...profile,
|
||||||
|
teams: assignedTeamIds
|
||||||
|
.map((teamId) => teamMap.get(teamId))
|
||||||
|
.filter(Boolean),
|
||||||
|
team_ids: assignedTeamIds,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function resolveTenantTeamIds(
|
||||||
|
server: FastifyInstance,
|
||||||
|
tenantId: number,
|
||||||
|
values: any[],
|
||||||
|
) {
|
||||||
|
const requestedTeamIds = normalizeTeamIds(values)
|
||||||
|
|
||||||
|
if (!requestedTeamIds.length) {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
|
||||||
|
const validTeams = await server.db
|
||||||
|
.select({ id: teams.id })
|
||||||
|
.from(teams)
|
||||||
|
.where(
|
||||||
|
and(
|
||||||
|
eq(teams.tenant, tenantId),
|
||||||
|
inArray(teams.id, requestedTeamIds)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
const validTeamIds = validTeams.map((team) => team.id)
|
||||||
|
|
||||||
|
if (validTeamIds.length !== requestedTeamIds.length) {
|
||||||
|
throw new Error("INVALID_TEAM_SELECTION")
|
||||||
|
}
|
||||||
|
|
||||||
|
return validTeamIds
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function syncProfileTeams(
|
||||||
|
server: FastifyInstance,
|
||||||
|
profileId: string,
|
||||||
|
teamIds: number[],
|
||||||
|
userId?: string | null
|
||||||
|
) {
|
||||||
|
await server.db
|
||||||
|
.delete(authProfileTeams)
|
||||||
|
.where(eq(authProfileTeams.profile_id, profileId))
|
||||||
|
|
||||||
|
if (!teamIds.length) return
|
||||||
|
|
||||||
|
await server.db
|
||||||
|
.insert(authProfileTeams)
|
||||||
|
.values(teamIds.map((teamId) => ({
|
||||||
|
profile_id: profileId,
|
||||||
|
team_id: teamId,
|
||||||
|
created_by: userId || null,
|
||||||
|
})))
|
||||||
|
}
|
||||||
@@ -36,6 +36,7 @@ import {
|
|||||||
spaces,
|
spaces,
|
||||||
statementallocations,
|
statementallocations,
|
||||||
tasks,
|
tasks,
|
||||||
|
teams,
|
||||||
texttemplates,
|
texttemplates,
|
||||||
units,
|
units,
|
||||||
vehicles,
|
vehicles,
|
||||||
@@ -171,6 +172,11 @@ export const resourceConfig = {
|
|||||||
searchColumns: ["name","number","description"],
|
searchColumns: ["name","number","description"],
|
||||||
numberRangeHolder: "number",
|
numberRangeHolder: "number",
|
||||||
},
|
},
|
||||||
|
teams: {
|
||||||
|
table: teams,
|
||||||
|
searchColumns: ["name", "description"],
|
||||||
|
mtoLoad: ["branch"],
|
||||||
|
},
|
||||||
tasks: {
|
tasks: {
|
||||||
table: tasks,
|
table: tasks,
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -63,6 +63,20 @@ const generateOldItemData = () => {
|
|||||||
}
|
}
|
||||||
generateOldItemData()
|
generateOldItemData()
|
||||||
|
|
||||||
|
const inputColumnCount = computed(() => {
|
||||||
|
return Array.isArray(dataType.inputColumns) && dataType.inputColumns.length > 0
|
||||||
|
? dataType.inputColumns.length
|
||||||
|
: 1
|
||||||
|
})
|
||||||
|
|
||||||
|
const getInputColumnStyle = () => {
|
||||||
|
if (props.platform === 'mobile') return undefined
|
||||||
|
|
||||||
|
return {
|
||||||
|
width: `${100 / inputColumnCount.value}%`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// --- ÄNDERUNG START: Computed Property statt Watcher/Function ---
|
// --- ÄNDERUNG START: Computed Property statt Watcher/Function ---
|
||||||
// Dies berechnet den Status automatisch neu, egal woher die Daten kommen (Init oder User-Eingabe)
|
// Dies berechnet den Status automatisch neu, egal woher die Daten kommen (Init oder User-Eingabe)
|
||||||
@@ -436,7 +450,8 @@ const updateItem = async () => {
|
|||||||
<div :class="platform === 'mobile' ?['flex','flex-col'] : ['flex','flex-row']">
|
<div :class="platform === 'mobile' ?['flex','flex-col'] : ['flex','flex-row']">
|
||||||
<div
|
<div
|
||||||
v-for="(columnName,index) in dataType.inputColumns"
|
v-for="(columnName,index) in dataType.inputColumns"
|
||||||
:class="platform === 'mobile' ? ['w-full'] : [`w-1/${dataType.inputColumns.length}`, ... index < dataType.inputColumns.length -1 ? ['mr-5'] : []]"
|
:class="platform === 'mobile' ? ['w-full'] : [... index < inputColumnCount - 1 ? ['mr-5'] : []]"
|
||||||
|
:style="getInputColumnStyle()"
|
||||||
>
|
>
|
||||||
<USeparator :label="columnName"/>
|
<USeparator :label="columnName"/>
|
||||||
|
|
||||||
|
|||||||
@@ -239,6 +239,11 @@ const links = computed(() => {
|
|||||||
to: "/standardEntity/branches",
|
to: "/standardEntity/branches",
|
||||||
icon: "i-heroicons-building-office-2"
|
icon: "i-heroicons-building-office-2"
|
||||||
} : null,
|
} : null,
|
||||||
|
featureEnabled("teams") ? {
|
||||||
|
label: "Teams",
|
||||||
|
to: "/standardEntity/teams",
|
||||||
|
icon: "i-heroicons-users"
|
||||||
|
} : null,
|
||||||
featureEnabled("staffProfiles") ? {
|
featureEnabled("staffProfiles") ? {
|
||||||
label: "Mitarbeiter",
|
label: "Mitarbeiter",
|
||||||
to: "/staff/profiles",
|
to: "/staff/profiles",
|
||||||
|
|||||||
@@ -244,6 +244,18 @@ export const useRole = () => {
|
|||||||
label: "Kostenstellen erstellen",
|
label: "Kostenstellen erstellen",
|
||||||
parent: "costcentres"
|
parent: "costcentres"
|
||||||
},
|
},
|
||||||
|
teams: {
|
||||||
|
label: "Teams",
|
||||||
|
showToAllUsers: false
|
||||||
|
},
|
||||||
|
"teams-viewAll": {
|
||||||
|
label: "Alle Teams einsehen",
|
||||||
|
parent: "teams"
|
||||||
|
},
|
||||||
|
"teams-create": {
|
||||||
|
label: "Teams erstellen",
|
||||||
|
parent: "teams"
|
||||||
|
},
|
||||||
ownaccounts: {
|
ownaccounts: {
|
||||||
label: "Buchungskonten",
|
label: "Buchungskonten",
|
||||||
showToAllUsers: false
|
showToAllUsers: false
|
||||||
|
|||||||
@@ -6,9 +6,11 @@ import resourceTimelinePlugin from "@fullcalendar/resource-timeline"
|
|||||||
import { parseDate } from "@internationalized/date"
|
import { parseDate } from "@internationalized/date"
|
||||||
|
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
|
const auth = useAuthStore()
|
||||||
const profileStore = useProfileStore()
|
const profileStore = useProfileStore()
|
||||||
const toast = useToast()
|
const toast = useToast()
|
||||||
const { $dayjs } = useNuxtApp()
|
const { $api, $dayjs } = useNuxtApp()
|
||||||
|
const { create: createEvent } = useEntities("events")
|
||||||
const { list: listStaffTimeSpans, createEntry, update: updateStaffTimeEntry } = useStaffTime()
|
const { list: listStaffTimeSpans, createEntry, update: updateStaffTimeEntry } = useStaffTime()
|
||||||
|
|
||||||
const loading = ref(true)
|
const loading = ref(true)
|
||||||
@@ -27,6 +29,21 @@ const resources = ref([])
|
|||||||
const events = ref([])
|
const events = ref([])
|
||||||
const profiles = ref([])
|
const profiles = ref([])
|
||||||
const inventoryitems = ref([])
|
const inventoryitems = ref([])
|
||||||
|
const savingQuickConfig = ref(false)
|
||||||
|
const quickEntryConfig = reactive({
|
||||||
|
name: "Quick-Eintrag",
|
||||||
|
color: "#2563eb"
|
||||||
|
})
|
||||||
|
const quickEntryColorOptions = [
|
||||||
|
{ label: "Blau", value: "#2563eb" },
|
||||||
|
{ label: "Grün", value: "#16a34a" },
|
||||||
|
{ label: "Orange", value: "#ea580c" },
|
||||||
|
{ label: "Rot", value: "#dc2626" },
|
||||||
|
{ label: "Pink", value: "#db2777" },
|
||||||
|
{ label: "Türkis", value: "#0f766e" },
|
||||||
|
{ label: "Grau", value: "#4b5563" },
|
||||||
|
{ label: "Schwarz", value: "#111827" }
|
||||||
|
]
|
||||||
|
|
||||||
const isAbsenceModalOpen = ref(false)
|
const isAbsenceModalOpen = ref(false)
|
||||||
const absenceForm = reactive({
|
const absenceForm = reactive({
|
||||||
@@ -75,6 +92,7 @@ const endDateValue = computed({
|
|||||||
|
|
||||||
const resourceTypeOptions = [
|
const resourceTypeOptions = [
|
||||||
{ label: "Alle Ressourcen", value: "all" },
|
{ label: "Alle Ressourcen", value: "all" },
|
||||||
|
{ label: "Teams", value: "Team" },
|
||||||
{ label: "Profile", value: "Profile" },
|
{ label: "Profile", value: "Profile" },
|
||||||
{ label: "Inventarartikel", value: "Inventarartikel" }
|
{ label: "Inventarartikel", value: "Inventarartikel" }
|
||||||
]
|
]
|
||||||
@@ -124,6 +142,15 @@ const absenceModalTitle = computed(() =>
|
|||||||
: (absenceForm.type === "sick" ? "Krank eintragen" : "Urlaub eintragen")
|
: (absenceForm.type === "sick" ? "Krank eintragen" : "Urlaub eintragen")
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const resolvedQuickEntryConfig = computed(() => {
|
||||||
|
const config = profileStore.ownTenant?.calendarConfig?.quickEntry || {}
|
||||||
|
|
||||||
|
return {
|
||||||
|
name: config.name || "Quick-Eintrag",
|
||||||
|
color: config.color || "#2563eb"
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
const visibleResources = computed(() => {
|
const visibleResources = computed(() => {
|
||||||
if (selectedType.value === "all") return resources.value
|
if (selectedType.value === "all") return resources.value
|
||||||
return resources.value.filter((resource) => resource.type === selectedType.value)
|
return resources.value.filter((resource) => resource.type === selectedType.value)
|
||||||
@@ -184,8 +211,7 @@ const calendarOptions = computed(() => ({
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
select(info) {
|
select(info) {
|
||||||
const resourceIds = info.resource?.id ? [info.resource.id] : []
|
createQuickEvent(info)
|
||||||
router.push(`/standardEntity/events/create?startDate=${encodeURIComponent(info.startStr)}&endDate=${encodeURIComponent(info.endStr)}&resources=${encodeURIComponent(JSON.stringify(resourceIds))}&source=timeline`)
|
|
||||||
},
|
},
|
||||||
eventClick(info) {
|
eventClick(info) {
|
||||||
if (info.event.extendedProps.entrytype === "staff-absence") {
|
if (info.event.extendedProps.entrytype === "staff-absence") {
|
||||||
@@ -199,7 +225,7 @@ const calendarOptions = computed(() => ({
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
router.push(`/standardEntity/events/show/${info.event.extendedProps.eventId}`)
|
router.push(`/standardEntity/events/edit/${info.event.extendedProps.eventId}`)
|
||||||
},
|
},
|
||||||
datesSet(info) {
|
datesSet(info) {
|
||||||
const nextFrom = $dayjs(info.start).format("YYYY-MM-DD")
|
const nextFrom = $dayjs(info.start).format("YYYY-MM-DD")
|
||||||
@@ -225,7 +251,12 @@ function resolveEventColor(eventType) {
|
|||||||
function resolveEventTitle(event, projectsById) {
|
function resolveEventTitle(event, projectsById) {
|
||||||
if (event.name) return event.name
|
if (event.name) return event.name
|
||||||
if (event.project && projectsById.has(event.project)) return projectsById.get(event.project).name
|
if (event.project && projectsById.has(event.project)) return projectsById.get(event.project).name
|
||||||
return "Planung"
|
return event.quick ? resolvedQuickEntryConfig.value.name : "Planung"
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveRenderedEventColor(event) {
|
||||||
|
if (event?.quick) return resolvedQuickEntryConfig.value.color
|
||||||
|
return resolveEventColor(event.eventtype)
|
||||||
}
|
}
|
||||||
|
|
||||||
function getProfileLabel(profile) {
|
function getProfileLabel(profile) {
|
||||||
@@ -241,23 +272,83 @@ function getAbsenceColor(type) {
|
|||||||
return type === "sick" ? "#dc2626" : "#d97706"
|
return type === "sick" ? "#dc2626" : "#d97706"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getTeamLabel(team) {
|
||||||
|
if (!team) return "Team"
|
||||||
|
return team.branch?.name ? `${team.name} (${team.branch.name})` : team.name
|
||||||
|
}
|
||||||
|
|
||||||
|
function getProfileResourceIds(profile) {
|
||||||
|
const assignedTeams = Array.isArray(profile?.teams) ? profile.teams.filter((team) => team?.id) : []
|
||||||
|
|
||||||
|
if (!assignedTeams.length) {
|
||||||
|
return [`P-${profile.id}`]
|
||||||
|
}
|
||||||
|
|
||||||
|
return assignedTeams.map((team) => `T-${team.id}:P-${profile.id}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeSelectedResourceIds(resourceId) {
|
||||||
|
if (!resourceId) return []
|
||||||
|
|
||||||
|
if (resourceId.startsWith("T-") && resourceId.includes(":P-")) {
|
||||||
|
return [`P-${resourceId.split(":P-")[1]}`]
|
||||||
|
}
|
||||||
|
|
||||||
|
if (resourceId.startsWith("T-")) {
|
||||||
|
const teamId = Number(resourceId.replace("T-", ""))
|
||||||
|
return profiles.value
|
||||||
|
.filter((profile) => (profile.teams || []).some((team) => team?.id === teamId))
|
||||||
|
.map((profile) => `P-${profile.id}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
return [resourceId]
|
||||||
|
}
|
||||||
|
|
||||||
function buildResources({ profiles, inventoryitems }) {
|
function buildResources({ profiles, inventoryitems }) {
|
||||||
return [
|
const teamResources = []
|
||||||
...profiles
|
const profileResources = []
|
||||||
|
|
||||||
|
profiles
|
||||||
.filter((profile) => !profile.archived)
|
.filter((profile) => !profile.archived)
|
||||||
.map((profile) => ({
|
.forEach((profile) => {
|
||||||
|
const assignedTeams = Array.isArray(profile?.teams) ? profile.teams.filter((team) => team?.id) : []
|
||||||
|
|
||||||
|
if (!assignedTeams.length) {
|
||||||
|
profileResources.push({
|
||||||
id: `P-${profile.id}`,
|
id: `P-${profile.id}`,
|
||||||
type: "Profile",
|
type: "Profile",
|
||||||
title: getProfileLabel(profile)
|
title: getProfileLabel(profile)
|
||||||
})),
|
})
|
||||||
...inventoryitems
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
assignedTeams.forEach((team) => {
|
||||||
|
if (!teamResources.find((resource) => resource.id === `T-${team.id}`)) {
|
||||||
|
teamResources.push({
|
||||||
|
id: `T-${team.id}`,
|
||||||
|
type: "Team",
|
||||||
|
title: getTeamLabel(team)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
profileResources.push({
|
||||||
|
id: `T-${team.id}:P-${profile.id}`,
|
||||||
|
parentId: `T-${team.id}`,
|
||||||
|
type: "Profile",
|
||||||
|
title: getProfileLabel(profile)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
const inventoryResources = inventoryitems
|
||||||
.filter((item) => !item.archived && item.usePlanning)
|
.filter((item) => !item.archived && item.usePlanning)
|
||||||
.map((item) => ({
|
.map((item) => ({
|
||||||
id: `I-${item.id}`,
|
id: `I-${item.id}`,
|
||||||
type: "Inventarartikel",
|
type: "Inventarartikel",
|
||||||
title: item.name
|
title: item.name
|
||||||
}))
|
}))
|
||||||
]
|
|
||||||
|
return [...teamResources, ...profileResources, ...inventoryResources]
|
||||||
}
|
}
|
||||||
|
|
||||||
function buildEvents({ rawEvents, projectsById }) {
|
function buildEvents({ rawEvents, projectsById }) {
|
||||||
@@ -265,7 +356,9 @@ function buildEvents({ rawEvents, projectsById }) {
|
|||||||
.filter((event) => !event.archived)
|
.filter((event) => !event.archived)
|
||||||
.map((event) => {
|
.map((event) => {
|
||||||
const resourceIds = [
|
const resourceIds = [
|
||||||
...(event.profiles || []).map((profileId) => `P-${profileId}`),
|
...(profiles.value
|
||||||
|
.filter((profile) => (event.profiles || []).includes(profile.id))
|
||||||
|
.flatMap((profile) => getProfileResourceIds(profile))),
|
||||||
...(event.inventoryitems || []).map((itemId) => `I-${itemId}`)
|
...(event.inventoryitems || []).map((itemId) => `I-${itemId}`)
|
||||||
]
|
]
|
||||||
|
|
||||||
@@ -274,8 +367,8 @@ function buildEvents({ rawEvents, projectsById }) {
|
|||||||
start: event.startDate,
|
start: event.startDate,
|
||||||
end: event.endDate,
|
end: event.endDate,
|
||||||
resourceIds,
|
resourceIds,
|
||||||
backgroundColor: resolveEventColor(event.eventtype),
|
backgroundColor: resolveRenderedEventColor(event),
|
||||||
borderColor: resolveEventColor(event.eventtype),
|
borderColor: resolveRenderedEventColor(event),
|
||||||
textColor: "#ffffff",
|
textColor: "#ffffff",
|
||||||
entrytype: "event",
|
entrytype: "event",
|
||||||
eventId: event.id
|
eventId: event.id
|
||||||
@@ -303,7 +396,7 @@ function buildAbsenceEvents(spansByUserId) {
|
|||||||
title: getAbsenceTitle(entry),
|
title: getAbsenceTitle(entry),
|
||||||
start: $dayjs(entry.started_at).startOf("day").toISOString(),
|
start: $dayjs(entry.started_at).startOf("day").toISOString(),
|
||||||
end: $dayjs(entry.stopped_at || entry.started_at).add(1, "day").startOf("day").toISOString(),
|
end: $dayjs(entry.stopped_at || entry.started_at).add(1, "day").startOf("day").toISOString(),
|
||||||
resourceIds: [`P-${profile.id}`],
|
resourceIds: getProfileResourceIds(profile),
|
||||||
allDay: true,
|
allDay: true,
|
||||||
backgroundColor: getAbsenceColor(entry.type),
|
backgroundColor: getAbsenceColor(entry.type),
|
||||||
borderColor: getAbsenceColor(entry.type),
|
borderColor: getAbsenceColor(entry.type),
|
||||||
@@ -350,6 +443,94 @@ function moveCalendarToday() {
|
|||||||
api.today()
|
api.today()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function openQuickConfig() {
|
||||||
|
quickEntryConfig.name = resolvedQuickEntryConfig.value.name
|
||||||
|
quickEntryConfig.color = resolvedQuickEntryConfig.value.color
|
||||||
|
}
|
||||||
|
|
||||||
|
async function saveQuickConfig() {
|
||||||
|
if (savingQuickConfig.value) return
|
||||||
|
|
||||||
|
const name = quickEntryConfig.name?.trim()
|
||||||
|
if (!name) {
|
||||||
|
toast.add({ title: "Name fehlt", description: "Bitte einen Namen für Quick-Einträge angeben.", color: "orange" })
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
savingQuickConfig.value = true
|
||||||
|
|
||||||
|
try {
|
||||||
|
const currentTenantData = auth.activeTenantData || {}
|
||||||
|
const nextCalendarConfig = {
|
||||||
|
...(currentTenantData.calendarConfig || {}),
|
||||||
|
quickEntry: {
|
||||||
|
name,
|
||||||
|
color: quickEntryConfig.color || "#2563eb"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const updatedTenant = await $api(`/api/tenant/other/${auth.activeTenant}`, {
|
||||||
|
method: "PUT",
|
||||||
|
body: {
|
||||||
|
data: {
|
||||||
|
calendarConfig: nextCalendarConfig
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
auth.activeTenantData = updatedTenant
|
||||||
|
profileStore.ownTenant = updatedTenant
|
||||||
|
toast.add({ title: "Quick-Einträge gespeichert", color: "green" })
|
||||||
|
await loadPlanningBoard()
|
||||||
|
} catch (error) {
|
||||||
|
console.error("saveQuickConfig failed", error)
|
||||||
|
toast.add({
|
||||||
|
title: "Quick-Konfiguration konnte nicht gespeichert werden",
|
||||||
|
description: error?.message || "Bitte erneut versuchen.",
|
||||||
|
color: "red"
|
||||||
|
})
|
||||||
|
} finally {
|
||||||
|
savingQuickConfig.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function createQuickEvent(info) {
|
||||||
|
const resourceIds = normalizeSelectedResourceIds(info.resource?.id)
|
||||||
|
|
||||||
|
const payload = {
|
||||||
|
name: resolvedQuickEntryConfig.value.name,
|
||||||
|
quick: true,
|
||||||
|
startDate: info.startStr,
|
||||||
|
endDate: info.endStr,
|
||||||
|
profiles: resourceIds
|
||||||
|
.filter((resourceId) => resourceId.startsWith("P-"))
|
||||||
|
.map((resourceId) => resourceId.replace("P-", "")),
|
||||||
|
inventoryitems: resourceIds
|
||||||
|
.filter((resourceId) => resourceId.startsWith("I-"))
|
||||||
|
.map((resourceId) => Number(resourceId.replace("I-", ""))),
|
||||||
|
inventoryitemgroups: [],
|
||||||
|
vehicles: [],
|
||||||
|
notes: "",
|
||||||
|
link: "",
|
||||||
|
project: null,
|
||||||
|
customer: null,
|
||||||
|
vendor: null,
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await createEvent(payload, true)
|
||||||
|
toast.add({ title: "Quick-Eintrag angelegt", color: "green" })
|
||||||
|
await loadPlanningBoard()
|
||||||
|
} catch (error) {
|
||||||
|
console.error("createQuickEvent failed", error)
|
||||||
|
toast.add({
|
||||||
|
title: "Quick-Eintrag konnte nicht angelegt werden",
|
||||||
|
description: error?.message || "Bitte erneut versuchen.",
|
||||||
|
color: "red"
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function openAbsenceModal(type = "vacation", preset = {}) {
|
function openAbsenceModal(type = "vacation", preset = {}) {
|
||||||
absenceForm.mode = preset.entry ? "edit" : "create"
|
absenceForm.mode = preset.entry ? "edit" : "create"
|
||||||
absenceForm.entry = preset.entry || null
|
absenceForm.entry = preset.entry || null
|
||||||
@@ -414,13 +595,15 @@ async function loadPlanningBoard() {
|
|||||||
loading.value = true
|
loading.value = true
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const [rawEvents, projects, profileRows, inventoryItemRows] = await Promise.all([
|
const [rawEvents, projects, profileResponse, inventoryItemRows] = await Promise.all([
|
||||||
useEntities("events").select(),
|
useEntities("events").select(),
|
||||||
useEntities("projects").select(),
|
useEntities("projects").select(),
|
||||||
useEntities("profiles").select(),
|
useNuxtApp().$api("/api/tenant/profiles"),
|
||||||
useEntities("inventoryitems").select()
|
useEntities("inventoryitems").select()
|
||||||
])
|
])
|
||||||
|
|
||||||
|
const profileRows = profileResponse?.data || []
|
||||||
|
|
||||||
const absenceSpansByUserId = await Promise.all(
|
const absenceSpansByUserId = await Promise.all(
|
||||||
(profileRows || [])
|
(profileRows || [])
|
||||||
.filter((profile) => profile.user_id)
|
.filter((profile) => profile.user_id)
|
||||||
@@ -537,6 +720,78 @@ onMounted(() => {
|
|||||||
</template>
|
</template>
|
||||||
<template #right>
|
<template #right>
|
||||||
<div class="flex flex-wrap items-center gap-2">
|
<div class="flex flex-wrap items-center gap-2">
|
||||||
|
<UPopover>
|
||||||
|
<UButton
|
||||||
|
color="neutral"
|
||||||
|
variant="outline"
|
||||||
|
icon="i-heroicons-cog-6-tooth"
|
||||||
|
@click="openQuickConfig"
|
||||||
|
>
|
||||||
|
Quick-Einträge
|
||||||
|
</UButton>
|
||||||
|
|
||||||
|
<template #content>
|
||||||
|
<div class="w-[320px] space-y-4 p-4">
|
||||||
|
<div>
|
||||||
|
<h3 class="text-sm font-semibold text-highlighted">Quick-Einträge</h3>
|
||||||
|
<p class="text-xs text-muted">
|
||||||
|
Name und Farbe für neu gezogene Quick-Einträge in der Plantafel.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<UFormField label="Name">
|
||||||
|
<UInput v-model="quickEntryConfig.name" class="w-full" />
|
||||||
|
</UFormField>
|
||||||
|
|
||||||
|
<UFormField label="Farbe">
|
||||||
|
<div class="space-y-3">
|
||||||
|
<div class="flex flex-wrap gap-2">
|
||||||
|
<button
|
||||||
|
v-for="option in quickEntryColorOptions"
|
||||||
|
:key="option.value"
|
||||||
|
type="button"
|
||||||
|
class="flex items-center gap-2 rounded border px-2 py-1 text-xs transition"
|
||||||
|
:class="quickEntryConfig.color === option.value ? 'border-primary ring-1 ring-primary' : 'border-default'"
|
||||||
|
@click="quickEntryConfig.color = option.value"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
class="h-4 w-4 rounded-full border border-white/40"
|
||||||
|
:style="{ backgroundColor: option.value }"
|
||||||
|
/>
|
||||||
|
{{ option.label }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="flex items-center gap-3">
|
||||||
|
<input
|
||||||
|
v-model="quickEntryConfig.color"
|
||||||
|
type="color"
|
||||||
|
class="h-10 w-14 cursor-pointer rounded border border-default bg-transparent p-1"
|
||||||
|
>
|
||||||
|
<UInput v-model="quickEntryConfig.color" class="flex-1" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</UFormField>
|
||||||
|
|
||||||
|
<div class="rounded border border-default p-3">
|
||||||
|
<p class="text-xs text-muted">Vorschau</p>
|
||||||
|
<div class="mt-2 inline-flex rounded px-3 py-1 text-sm font-medium text-white" :style="{ backgroundColor: quickEntryConfig.color }">
|
||||||
|
{{ quickEntryConfig.name || "Quick-Eintrag" }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="flex justify-end">
|
||||||
|
<UButton
|
||||||
|
color="primary"
|
||||||
|
:loading="savingQuickConfig"
|
||||||
|
@click="saveQuickConfig"
|
||||||
|
>
|
||||||
|
Speichern
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</UPopover>
|
||||||
<UButton
|
<UButton
|
||||||
color="amber"
|
color="amber"
|
||||||
variant="soft"
|
variant="soft"
|
||||||
|
|||||||
@@ -36,6 +36,7 @@ const defaultFeatures = {
|
|||||||
incomingInvoices: true,
|
incomingInvoices: true,
|
||||||
costcentres: true,
|
costcentres: true,
|
||||||
branches: true,
|
branches: true,
|
||||||
|
teams: true,
|
||||||
accounts: true,
|
accounts: true,
|
||||||
ownaccounts: true,
|
ownaccounts: true,
|
||||||
banking: true,
|
banking: true,
|
||||||
@@ -82,6 +83,7 @@ const featureOptions = [
|
|||||||
{ key: "incomingInvoices", label: "Buchhaltung: Eingangsbelege" },
|
{ key: "incomingInvoices", label: "Buchhaltung: Eingangsbelege" },
|
||||||
{ key: "costcentres", label: "Buchhaltung: Kostenstellen" },
|
{ key: "costcentres", label: "Buchhaltung: Kostenstellen" },
|
||||||
{ key: "branches", label: "Stammdaten: Niederlassungen" },
|
{ key: "branches", label: "Stammdaten: Niederlassungen" },
|
||||||
|
{ key: "teams", label: "Stammdaten: Teams" },
|
||||||
{ key: "accounts", label: "Buchhaltung: Buchungskonten" },
|
{ key: "accounts", label: "Buchhaltung: Buchungskonten" },
|
||||||
{ key: "ownaccounts", label: "Buchhaltung: Zusätzliche Buchungskonten" },
|
{ key: "ownaccounts", label: "Buchhaltung: Zusätzliche Buchungskonten" },
|
||||||
{ key: "banking", label: "Buchhaltung: Bank" },
|
{ key: "banking", label: "Buchhaltung: Bank" },
|
||||||
|
|||||||
@@ -7,8 +7,13 @@ const { $api } = useNuxtApp()
|
|||||||
const id = route.params.id as string
|
const id = route.params.id as string
|
||||||
const profile = ref<any>(null)
|
const profile = ref<any>(null)
|
||||||
const branches = ref<any[]>([])
|
const branches = ref<any[]>([])
|
||||||
|
const teams = ref<any[]>([])
|
||||||
const pending = ref(true)
|
const pending = ref(true)
|
||||||
const saving = ref(false)
|
const saving = ref(false)
|
||||||
|
const selectMenuUi = {
|
||||||
|
base: 'w-full',
|
||||||
|
content: 'min-w-[min(32rem,90vw)] w-max max-w-[90vw]'
|
||||||
|
}
|
||||||
|
|
||||||
async function fetchBranches() {
|
async function fetchBranches() {
|
||||||
try {
|
try {
|
||||||
@@ -19,6 +24,15 @@ async function fetchBranches() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function fetchTeams() {
|
||||||
|
try {
|
||||||
|
teams.value = await useEntities("teams").select()
|
||||||
|
} catch (err) {
|
||||||
|
console.error('[fetchTeams]', err)
|
||||||
|
teams.value = []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/** Profil laden **/
|
/** Profil laden **/
|
||||||
async function fetchProfile() {
|
async function fetchProfile() {
|
||||||
pending.value = true
|
pending.value = true
|
||||||
@@ -26,6 +40,7 @@ async function fetchProfile() {
|
|||||||
profile.value = await $api(`/api/profiles/${id}`)
|
profile.value = await $api(`/api/profiles/${id}`)
|
||||||
ensureWorkingHoursStructure()
|
ensureWorkingHoursStructure()
|
||||||
ensureBranchStructure()
|
ensureBranchStructure()
|
||||||
|
ensureTeamStructure()
|
||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
console.error('[fetchProfile]', err)
|
console.error('[fetchProfile]', err)
|
||||||
toast.add({
|
toast.add({
|
||||||
@@ -58,6 +73,20 @@ function ensureBranchStructure() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function ensureTeamStructure() {
|
||||||
|
if (!profile.value) return
|
||||||
|
|
||||||
|
if (!Array.isArray(profile.value.team_ids)) {
|
||||||
|
if (Array.isArray(profile.value.teams)) {
|
||||||
|
profile.value.team_ids = profile.value.teams
|
||||||
|
.map((entry: any) => entry?.id ?? entry)
|
||||||
|
.filter((entry: any) => entry != null)
|
||||||
|
} else {
|
||||||
|
profile.value.team_ids = []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const updatePrimaryBranch = (value: number | null) => {
|
const updatePrimaryBranch = (value: number | null) => {
|
||||||
if (!profile.value) return
|
if (!profile.value) return
|
||||||
profile.value.branch_id = value
|
profile.value.branch_id = value
|
||||||
@@ -77,6 +106,11 @@ const updateBranchMemberships = (values: number[]) => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const updateTeamMemberships = (values: number[]) => {
|
||||||
|
if (!profile.value) return
|
||||||
|
profile.value.team_ids = values || []
|
||||||
|
}
|
||||||
|
|
||||||
/** Profil speichern **/
|
/** Profil speichern **/
|
||||||
async function saveProfile() {
|
async function saveProfile() {
|
||||||
if (saving.value) return
|
if (saving.value) return
|
||||||
@@ -180,7 +214,7 @@ const checkZip = async () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
onMounted(async () => {
|
onMounted(async () => {
|
||||||
await Promise.all([fetchBranches(), fetchProfile()])
|
await Promise.all([fetchBranches(), fetchTeams(), fetchProfile()])
|
||||||
})
|
})
|
||||||
|
|
||||||
</script>
|
</script>
|
||||||
@@ -234,29 +268,29 @@ onMounted(async () => {
|
|||||||
<USeparator label="Persönliche Daten" />
|
<USeparator label="Persönliche Daten" />
|
||||||
|
|
||||||
<UForm :state="profile" @submit.prevent="saveProfile" class="grid grid-cols-1 md:grid-cols-2 gap-6 mt-4">
|
<UForm :state="profile" @submit.prevent="saveProfile" class="grid grid-cols-1 md:grid-cols-2 gap-6 mt-4">
|
||||||
<UFormField label="Vorname">
|
<UFormField label="Vorname" class="w-full">
|
||||||
<UInput v-model="profile.first_name" />
|
<UInput v-model="profile.first_name" class="w-full" />
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Nachname">
|
<UFormField label="Nachname" class="w-full">
|
||||||
<UInput v-model="profile.last_name" />
|
<UInput v-model="profile.last_name" class="w-full" />
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="E-Mail">
|
<UFormField label="E-Mail" class="w-full">
|
||||||
<UInput v-model="profile.email" />
|
<UInput v-model="profile.email" class="w-full" />
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Telefon (Mobil)">
|
<UFormField label="Telefon (Mobil)" class="w-full">
|
||||||
<UInput v-model="profile.mobile_tel" />
|
<UInput v-model="profile.mobile_tel" class="w-full" />
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Telefon (Festnetz)">
|
<UFormField label="Telefon (Festnetz)" class="w-full">
|
||||||
<UInput v-model="profile.fixed_tel" />
|
<UInput v-model="profile.fixed_tel" class="w-full" />
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Geburtstag">
|
<UFormField label="Geburtstag" class="w-full">
|
||||||
<div class="flex items-center gap-2">
|
<div class="flex items-center gap-2">
|
||||||
<UInput type="date" v-model="profile.birthday" class="flex-1" />
|
<UInput type="date" v-model="profile.birthday" class="flex-1 w-full" />
|
||||||
<UButton color="gray" variant="soft" label="Heute" @click="setProfileDate('birthday')" />
|
<UButton color="gray" variant="soft" label="Heute" @click="setProfileDate('birthday')" />
|
||||||
</div>
|
</div>
|
||||||
</UFormField>
|
</UFormField>
|
||||||
@@ -266,37 +300,37 @@ onMounted(async () => {
|
|||||||
<USeparator label="Vertragsinformationen" />
|
<USeparator label="Vertragsinformationen" />
|
||||||
|
|
||||||
<UForm :state="profile" @submit.prevent="saveProfile" class="grid grid-cols-1 md:grid-cols-2 gap-6 mt-4">
|
<UForm :state="profile" @submit.prevent="saveProfile" class="grid grid-cols-1 md:grid-cols-2 gap-6 mt-4">
|
||||||
<UFormField label="Vertragsart">
|
<UFormField label="Vertragsart" class="w-full">
|
||||||
<UInput v-model="profile.contract_type"/>
|
<UInput v-model="profile.contract_type" class="w-full"/>
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Status">
|
<UFormField label="Status" class="w-full">
|
||||||
<UInput v-model="profile.status"/>
|
<UInput v-model="profile.status" class="w-full"/>
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Position">
|
<UFormField label="Position" class="w-full">
|
||||||
<UInput v-model="profile.position"/>
|
<UInput v-model="profile.position" class="w-full"/>
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Qualifikation">
|
<UFormField label="Qualifikation" class="w-full">
|
||||||
<UInput v-model="profile.qualification"/>
|
<UInput v-model="profile.qualification" class="w-full"/>
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Eintrittsdatum">
|
<UFormField label="Eintrittsdatum" class="w-full">
|
||||||
<div class="flex items-center gap-2">
|
<div class="flex items-center gap-2">
|
||||||
<UInput type="date" v-model="profile.entry_date" class="flex-1" />
|
<UInput type="date" v-model="profile.entry_date" class="flex-1 w-full" />
|
||||||
<UButton color="gray" variant="soft" label="Heute" @click="setProfileDate('entry_date')" />
|
<UButton color="gray" variant="soft" label="Heute" @click="setProfileDate('entry_date')" />
|
||||||
</div>
|
</div>
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Wöchentliche Arbeitszeit (Std)">
|
<UFormField label="Wöchentliche Arbeitszeit (Std)" class="w-full">
|
||||||
<UInput type="number" v-model="profile.weekly_working_hours" />
|
<UInput type="number" v-model="profile.weekly_working_hours" class="w-full" />
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Bezahlte Urlaubstage (Jahr)">
|
<UFormField label="Bezahlte Urlaubstage (Jahr)" class="w-full">
|
||||||
<UInput type="number" v-model="profile.annual_paid_leave_days" />
|
<UInput type="number" v-model="profile.annual_paid_leave_days" class="w-full" />
|
||||||
</UFormField>
|
</UFormField>
|
||||||
<UFormField label="Aktiv">
|
<UFormField label="Aktiv" class="w-full">
|
||||||
<div class="flex items-center gap-3">
|
<div class="flex items-center gap-3">
|
||||||
<USwitch v-model="profile.active" color="primary" />
|
<USwitch v-model="profile.active" color="primary" />
|
||||||
<span class="text-sm text-gray-600">
|
<span class="text-sm text-gray-600">
|
||||||
@@ -311,52 +345,83 @@ onMounted(async () => {
|
|||||||
<USeparator label="Niederlassungen" />
|
<USeparator label="Niederlassungen" />
|
||||||
|
|
||||||
<UForm :state="profile" @submit.prevent="saveProfile" class="grid grid-cols-1 md:grid-cols-2 gap-6 mt-4">
|
<UForm :state="profile" @submit.prevent="saveProfile" class="grid grid-cols-1 md:grid-cols-2 gap-6 mt-4">
|
||||||
<UFormField label="Primäre Niederlassung">
|
<UFormField label="Primäre Niederlassung" class="w-full">
|
||||||
<USelectMenu
|
<USelectMenu
|
||||||
:model-value="profile.branch_id"
|
:model-value="profile.branch_id"
|
||||||
:items="branches"
|
:items="branches"
|
||||||
label-key="name"
|
label-key="name"
|
||||||
value-key="id"
|
value-key="id"
|
||||||
|
class="w-full"
|
||||||
|
:ui="selectMenuUi"
|
||||||
@update:model-value="updatePrimaryBranch"
|
@update:model-value="updatePrimaryBranch"
|
||||||
/>
|
/>
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Weitere Niederlassungen">
|
<UFormField label="Weitere Niederlassungen" class="w-full">
|
||||||
<USelectMenu
|
<USelectMenu
|
||||||
:model-value="profile.branch_ids"
|
:model-value="profile.branch_ids"
|
||||||
:items="branches"
|
:items="branches"
|
||||||
label-key="name"
|
label-key="name"
|
||||||
value-key="id"
|
value-key="id"
|
||||||
multiple
|
multiple
|
||||||
|
class="w-full"
|
||||||
|
:ui="selectMenuUi"
|
||||||
@update:model-value="updateBranchMemberships"
|
@update:model-value="updateBranchMemberships"
|
||||||
/>
|
/>
|
||||||
</UFormField>
|
</UFormField>
|
||||||
</UForm>
|
</UForm>
|
||||||
</UCard>
|
</UCard>
|
||||||
|
|
||||||
|
<UCard v-if="!pending && profile" class="mt-3">
|
||||||
|
<USeparator label="Teams" />
|
||||||
|
|
||||||
|
<UForm :state="profile" @submit.prevent="saveProfile" class="grid grid-cols-1 md:grid-cols-2 gap-6 mt-4">
|
||||||
|
<UFormField label="Team-Zuordnung" class="w-full">
|
||||||
|
<USelectMenu
|
||||||
|
:model-value="profile.team_ids"
|
||||||
|
:items="teams"
|
||||||
|
label-key="name"
|
||||||
|
value-key="id"
|
||||||
|
multiple
|
||||||
|
class="w-full"
|
||||||
|
:ui="selectMenuUi"
|
||||||
|
@update:model-value="updateTeamMemberships"
|
||||||
|
/>
|
||||||
|
</UFormField>
|
||||||
|
|
||||||
|
<UFormField label="Hinweis" class="w-full">
|
||||||
|
<div class="text-sm text-gray-500 pt-2">
|
||||||
|
Teams können in den Stammdaten gepflegt und optional einer Niederlassung zugeordnet werden.
|
||||||
|
</div>
|
||||||
|
</UFormField>
|
||||||
|
</UForm>
|
||||||
|
</UCard>
|
||||||
|
|
||||||
<UCard v-if="!pending && profile" class="mt-3">
|
<UCard v-if="!pending && profile" class="mt-3">
|
||||||
<USeparator label="Adresse & Standort" />
|
<USeparator label="Adresse & Standort" />
|
||||||
|
|
||||||
<UForm :state="profile" @submit.prevent="saveProfile" class="grid grid-cols-1 md:grid-cols-2 gap-6 mt-4">
|
<UForm :state="profile" @submit.prevent="saveProfile" class="grid grid-cols-1 md:grid-cols-2 gap-6 mt-4">
|
||||||
<UFormField label="Straße und Hausnummer">
|
<UFormField label="Straße und Hausnummer" class="w-full">
|
||||||
<UInput v-model="profile.address_street"/>
|
<UInput v-model="profile.address_street" class="w-full"/>
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="PLZ">
|
<UFormField label="PLZ" class="w-full">
|
||||||
<UInput type="text" v-model="profile.address_zip" @focusout="checkZip"/>
|
<UInput type="text" v-model="profile.address_zip" class="w-full" @focusout="checkZip"/>
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Ort">
|
<UFormField label="Ort" class="w-full">
|
||||||
<UInput v-model="profile.address_city"/>
|
<UInput v-model="profile.address_city" class="w-full"/>
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Bundesland">
|
<UFormField label="Bundesland" class="w-full">
|
||||||
<USelectMenu
|
<USelectMenu
|
||||||
v-model="profile.state_code"
|
v-model="profile.state_code"
|
||||||
:options="bundeslaender"
|
:options="bundeslaender"
|
||||||
value-attribute="code"
|
value-attribute="code"
|
||||||
option-attribute="name"
|
option-attribute="name"
|
||||||
placeholder="Bundesland auswählen"
|
placeholder="Bundesland auswählen"
|
||||||
|
class="w-full"
|
||||||
|
:ui="selectMenuUi"
|
||||||
/>
|
/>
|
||||||
</UFormField>
|
</UFormField>
|
||||||
</UForm>
|
</UForm>
|
||||||
@@ -394,20 +459,20 @@ onMounted(async () => {
|
|||||||
<UCard v-if="!pending && profile" class="mt-3">
|
<UCard v-if="!pending && profile" class="mt-3">
|
||||||
<USeparator label="Sonstiges" />
|
<USeparator label="Sonstiges" />
|
||||||
<UForm :state="profile" @submit.prevent="saveProfile" class="grid grid-cols-1 md:grid-cols-2 gap-6 mt-4">
|
<UForm :state="profile" @submit.prevent="saveProfile" class="grid grid-cols-1 md:grid-cols-2 gap-6 mt-4">
|
||||||
<UFormField label="Kleidergröße (Oberteil)">
|
<UFormField label="Kleidergröße (Oberteil)" class="w-full">
|
||||||
<UInput v-model="profile.clothing_size_top" />
|
<UInput v-model="profile.clothing_size_top" class="w-full" />
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Kleidergröße (Hose)">
|
<UFormField label="Kleidergröße (Hose)" class="w-full">
|
||||||
<UInput v-model="profile.clothing_size_bottom" />
|
<UInput v-model="profile.clothing_size_bottom" class="w-full" />
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Schuhgröße">
|
<UFormField label="Schuhgröße" class="w-full">
|
||||||
<UInput v-model="profile.clothing_size_shoe" />
|
<UInput v-model="profile.clothing_size_shoe" class="w-full" />
|
||||||
</UFormField>
|
</UFormField>
|
||||||
|
|
||||||
<UFormField label="Token-ID">
|
<UFormField label="Token-ID" class="w-full">
|
||||||
<UInput v-model="profile.token_id" />
|
<UInput v-model="profile.token_id" class="w-full" />
|
||||||
</UFormField>
|
</UFormField>
|
||||||
</UForm>
|
</UForm>
|
||||||
</UCard>
|
</UCard>
|
||||||
|
|||||||
@@ -13,7 +13,8 @@
|
|||||||
employee_number: profile?.employee_number || '',
|
employee_number: profile?.employee_number || '',
|
||||||
full_name: profile?.full_name || user?.full_name || user?.email || 'Ohne Profil',
|
full_name: profile?.full_name || user?.full_name || user?.email || 'Ohne Profil',
|
||||||
email: user?.email || profile?.email || '',
|
email: user?.email || profile?.email || '',
|
||||||
branch_name: profile?.branch?.name || ''
|
branch_name: profile?.branch?.name || '',
|
||||||
|
team_names: (profile?.teams || []).map((team) => team?.name).filter(Boolean).join(', ')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -52,6 +53,9 @@
|
|||||||
},{
|
},{
|
||||||
key: "branch_name",
|
key: "branch_name",
|
||||||
label: "Niederlassung",
|
label: "Niederlassung",
|
||||||
|
},{
|
||||||
|
key: "team_names",
|
||||||
|
label: "Teams",
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
const selectedColumns = ref(templateColumns)
|
const selectedColumns = ref(templateColumns)
|
||||||
|
|||||||
@@ -2897,6 +2897,12 @@ export const useDataStore = defineStore('data', () => {
|
|||||||
inputType: "text",
|
inputType: "text",
|
||||||
sortable: true
|
sortable: true
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
key: "quick",
|
||||||
|
label: "Quick-Eintrag",
|
||||||
|
inputType: "bool",
|
||||||
|
sortable: true
|
||||||
|
},
|
||||||
{
|
{
|
||||||
key: "startDate",
|
key: "startDate",
|
||||||
label: "Start",
|
label: "Start",
|
||||||
@@ -3044,6 +3050,56 @@ export const useDataStore = defineStore('data', () => {
|
|||||||
],
|
],
|
||||||
showTabs: [{label: 'Informationen'},{label: 'Wiki'}]
|
showTabs: [{label: 'Informationen'},{label: 'Wiki'}]
|
||||||
},
|
},
|
||||||
|
teams: {
|
||||||
|
isArchivable: true,
|
||||||
|
label: "Teams",
|
||||||
|
labelSingle: "Team",
|
||||||
|
isStandardEntity: true,
|
||||||
|
redirect: true,
|
||||||
|
historyItemHolder: "team",
|
||||||
|
sortColumn: "name",
|
||||||
|
selectWithInformation: "*",
|
||||||
|
filters: [{
|
||||||
|
name: "Archivierte ausblenden",
|
||||||
|
default: true,
|
||||||
|
"filterFunction": function (row) {
|
||||||
|
if(!row.archived) {
|
||||||
|
return true
|
||||||
|
} else {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}],
|
||||||
|
inputColumns: ["Allgemein"],
|
||||||
|
templateColumns: [
|
||||||
|
{
|
||||||
|
key: "name",
|
||||||
|
label: "Name",
|
||||||
|
required: true,
|
||||||
|
title: true,
|
||||||
|
inputType: "text",
|
||||||
|
inputColumn: "Allgemein",
|
||||||
|
sortable: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "branch",
|
||||||
|
label: "Niederlassung",
|
||||||
|
inputType: "select",
|
||||||
|
inputColumn: "Allgemein",
|
||||||
|
selectDataType: "branches",
|
||||||
|
selectOptionAttribute: "name",
|
||||||
|
component: branch,
|
||||||
|
sortable: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "description",
|
||||||
|
label: "Beschreibung",
|
||||||
|
inputType: "textarea",
|
||||||
|
inputColumn: "Allgemein"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
showTabs: [{label: 'Informationen'},{label: 'Wiki'}]
|
||||||
|
},
|
||||||
workingtimes: {
|
workingtimes: {
|
||||||
isArchivable: true,
|
isArchivable: true,
|
||||||
label: "Anwesenheiten",
|
label: "Anwesenheiten",
|
||||||
|
|||||||
Reference in New Issue
Block a user