- PWA manifest on the NFC kiosk page so it installs as a full-screen home-screen app (Chrome "Install" / Safari "Add to Home Screen"). - Dedicated "Kiosk Operator" permission + gated "Fusion Clock Kiosk" top-level app (act_url -> /fusion_clock/kiosk/nfc). Kiosk controllers accept Manager OR Kiosk Operator; all kiosk data ops already run sudo. - Fix 403: read the company kiosk location via sudo on page-load and tap (Kiosk Operator has no fusion.clock.location ACL). - Odoo 19 permissions UX: ir.module.category + res.groups.privilege so User/Team Lead/Manager and Kiosk Operator appear as application-access dropdowns on the user form (no developer mode). Short group display names. - Docs: note res.groups.privilege as the Odoo 19 category_id replacement. Deployed live to entech (odoo-entech / LXC 111 on pve-worker5). v19.0.3.6.0. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
385 lines
20 KiB
XML
385 lines
20 KiB
XML
<?xml version="1.0" encoding="utf-8"?>
|
|
<odoo>
|
|
|
|
<!-- ================================================================
|
|
App category + privileges (Odoo 19) so Fusion Clock roles appear
|
|
as selectable application-access dropdowns on the user form,
|
|
exactly like the other Fusion apps (no developer mode needed).
|
|
Odoo 19 dropped res.groups.category_id; groups link to a
|
|
res.groups.privilege, which carries the category_id.
|
|
================================================================ -->
|
|
<record id="module_category_fusion_clock" model="ir.module.category">
|
|
<field name="name">Fusion Clock</field>
|
|
<field name="sequence">45</field>
|
|
</record>
|
|
|
|
<!-- Main role hierarchy (User < Team Lead < Manager) -> one dropdown -->
|
|
<record id="res_groups_privilege_fusion_clock" model="res.groups.privilege">
|
|
<field name="name">Fusion Clock</field>
|
|
<field name="sequence">45</field>
|
|
<field name="category_id" ref="module_category_fusion_clock"/>
|
|
</record>
|
|
|
|
<!-- Standalone kiosk-operator role -> its own row under the same header -->
|
|
<record id="res_groups_privilege_fusion_clock_kiosk" model="res.groups.privilege">
|
|
<field name="name">Fusion Clock Kiosk</field>
|
|
<field name="sequence">46</field>
|
|
<field name="category_id" ref="module_category_fusion_clock"/>
|
|
</record>
|
|
|
|
<!-- Groups -->
|
|
<record id="group_fusion_clock_user" model="res.groups">
|
|
<field name="name">User</field>
|
|
<field name="privilege_id" ref="res_groups_privilege_fusion_clock"/>
|
|
<field name="implied_ids" eval="[(4, ref('base.group_user'))]"/>
|
|
<field name="comment">Can clock in/out and view own attendance</field>
|
|
</record>
|
|
|
|
<record id="group_fusion_clock_team_lead" model="res.groups">
|
|
<field name="name">Team Lead</field>
|
|
<field name="privilege_id" ref="res_groups_privilege_fusion_clock"/>
|
|
<field name="implied_ids" eval="[(4, ref('group_fusion_clock_user'))]"/>
|
|
<field name="comment">Can view direct reports attendance (read-only)</field>
|
|
</record>
|
|
|
|
<record id="group_fusion_clock_manager" model="res.groups">
|
|
<field name="name">Manager</field>
|
|
<field name="privilege_id" ref="res_groups_privilege_fusion_clock"/>
|
|
<field name="implied_ids" eval="[(4, ref('group_fusion_clock_team_lead'))]"/>
|
|
<field name="comment">Can manage locations, view all attendance, generate reports</field>
|
|
</record>
|
|
|
|
<!-- Dedicated kiosk-operator permission: can run the shared clock kiosk
|
|
(NFC tap / PIN) WITHOUT full Clock Manager access. Gates the
|
|
"Fusion Clock Kiosk" app menu and is accepted by the kiosk controllers.
|
|
Implies only base.group_user, so it does NOT reveal the full Fusion
|
|
Clock app (which is gated to group_fusion_clock_user). -->
|
|
<record id="group_fusion_clock_kiosk_app" model="res.groups">
|
|
<field name="name">Kiosk Operator</field>
|
|
<field name="privilege_id" ref="res_groups_privilege_fusion_clock_kiosk"/>
|
|
<field name="implied_ids" eval="[(4, ref('base.group_user'))]"/>
|
|
<field name="comment">Can open and operate the shared clock kiosk (NFC tap / PIN) without full Clock Manager access. Intended for shared wall-tablet accounts.</field>
|
|
</record>
|
|
|
|
<!-- Auto-assign admin to Manager group -->
|
|
<function model="res.users" name="write">
|
|
<value eval="[ref('base.user_admin')]"/>
|
|
<value eval="{'group_ids': [(4, ref('group_fusion_clock_manager'))]}"/>
|
|
</function>
|
|
|
|
<!-- ================================================================
|
|
Record Rules - Clock Location
|
|
================================================================ -->
|
|
<record id="rule_clock_location_user" model="ir.rule">
|
|
<field name="name">Clock Location: User sees active company locations</field>
|
|
<field name="model_id" ref="model_fusion_clock_location"/>
|
|
<field name="domain_force">[('company_id', 'in', company_ids), ('active', '=', True)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_user'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_clock_location_manager" model="ir.rule">
|
|
<field name="name">Clock Location: Manager full access</field>
|
|
<field name="model_id" ref="model_fusion_clock_location"/>
|
|
<field name="domain_force">[('company_id', 'in', company_ids)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_manager'))]"/>
|
|
</record>
|
|
|
|
<!-- ================================================================
|
|
Record Rules - Clock Penalty
|
|
================================================================ -->
|
|
<record id="rule_clock_penalty_user" model="ir.rule">
|
|
<field name="name">Clock Penalty: User sees own penalties</field>
|
|
<field name="model_id" ref="model_fusion_clock_penalty"/>
|
|
<field name="domain_force">[('employee_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_user'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_clock_penalty_team_lead" model="ir.rule">
|
|
<field name="name">Clock Penalty: Team Lead sees direct reports</field>
|
|
<field name="model_id" ref="model_fusion_clock_penalty"/>
|
|
<field name="domain_force">['|', ('employee_id.user_id', '=', user.id), ('employee_id.parent_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_team_lead'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_clock_penalty_manager" model="ir.rule">
|
|
<field name="name">Clock Penalty: Manager full access</field>
|
|
<field name="model_id" ref="model_fusion_clock_penalty"/>
|
|
<field name="domain_force">[(1, '=', 1)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_manager'))]"/>
|
|
</record>
|
|
|
|
<!-- ================================================================
|
|
Record Rules - Clock Report
|
|
================================================================ -->
|
|
<record id="rule_clock_report_user" model="ir.rule">
|
|
<field name="name">Clock Report: User sees own reports</field>
|
|
<field name="model_id" ref="model_fusion_clock_report"/>
|
|
<field name="domain_force">['|', ('employee_id.user_id', '=', user.id), ('employee_id', '=', False)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_user'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_clock_report_manager" model="ir.rule">
|
|
<field name="name">Clock Report: Manager full access</field>
|
|
<field name="model_id" ref="model_fusion_clock_report"/>
|
|
<field name="domain_force">[(1, '=', 1)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_manager'))]"/>
|
|
</record>
|
|
|
|
<!-- ================================================================
|
|
Record Rules - Activity Log
|
|
================================================================ -->
|
|
<record id="rule_activity_log_user" model="ir.rule">
|
|
<field name="name">Activity Log: User sees own logs</field>
|
|
<field name="model_id" ref="model_fusion_clock_activity_log"/>
|
|
<field name="domain_force">[('employee_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_user'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_activity_log_team_lead" model="ir.rule">
|
|
<field name="name">Activity Log: Team Lead sees direct reports</field>
|
|
<field name="model_id" ref="model_fusion_clock_activity_log"/>
|
|
<field name="domain_force">['|', ('employee_id.user_id', '=', user.id), ('employee_id.parent_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_team_lead'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_activity_log_manager" model="ir.rule">
|
|
<field name="name">Activity Log: Manager full access</field>
|
|
<field name="model_id" ref="model_fusion_clock_activity_log"/>
|
|
<field name="domain_force">[(1, '=', 1)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_manager'))]"/>
|
|
</record>
|
|
|
|
<!-- ================================================================
|
|
Record Rules - Leave Request
|
|
================================================================ -->
|
|
<record id="rule_leave_request_user" model="ir.rule">
|
|
<field name="name">Leave Request: User sees own</field>
|
|
<field name="model_id" ref="model_fusion_clock_leave_request"/>
|
|
<field name="domain_force">[('employee_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_user'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_leave_request_manager" model="ir.rule">
|
|
<field name="name">Leave Request: Manager full access</field>
|
|
<field name="model_id" ref="model_fusion_clock_leave_request"/>
|
|
<field name="domain_force">[(1, '=', 1)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_manager'))]"/>
|
|
</record>
|
|
|
|
<!-- ================================================================
|
|
Record Rules - Shift
|
|
================================================================ -->
|
|
<record id="rule_shift_user" model="ir.rule">
|
|
<field name="name">Shift: User reads active</field>
|
|
<field name="model_id" ref="model_fusion_clock_shift"/>
|
|
<field name="domain_force">[('active', '=', True)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_user'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_shift_manager" model="ir.rule">
|
|
<field name="name">Shift: Manager full access</field>
|
|
<field name="model_id" ref="model_fusion_clock_shift"/>
|
|
<field name="domain_force">[(1, '=', 1)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_manager'))]"/>
|
|
</record>
|
|
|
|
<!-- ================================================================
|
|
Record Rules - Dated Schedules
|
|
================================================================ -->
|
|
<record id="rule_schedule_user" model="ir.rule">
|
|
<field name="name">Schedule: User sees own</field>
|
|
<field name="model_id" ref="model_fusion_clock_schedule"/>
|
|
<field name="domain_force">[('employee_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_user'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_schedule_team_lead" model="ir.rule">
|
|
<field name="name">Schedule: Team Lead sees direct reports</field>
|
|
<field name="model_id" ref="model_fusion_clock_schedule"/>
|
|
<field name="domain_force">['|', ('employee_id.user_id', '=', user.id), ('employee_id.parent_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_team_lead'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_schedule_manager" model="ir.rule">
|
|
<field name="name">Schedule: Manager full access</field>
|
|
<field name="model_id" ref="model_fusion_clock_schedule"/>
|
|
<field name="domain_force">[('company_id', 'in', company_ids)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_manager'))]"/>
|
|
</record>
|
|
|
|
<record id="rule_schedule_audit_manager" model="ir.rule">
|
|
<field name="name">Schedule Audit: Manager reads all</field>
|
|
<field name="model_id" ref="model_fusion_clock_schedule_audit"/>
|
|
<field name="domain_force">[('company_id', 'in', company_ids)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_manager'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<!-- ================================================================
|
|
Record Rules - Correction Request
|
|
================================================================ -->
|
|
<record id="rule_correction_user" model="ir.rule">
|
|
<field name="name">Correction: User sees own</field>
|
|
<field name="model_id" ref="model_fusion_clock_correction"/>
|
|
<field name="domain_force">[('employee_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_user'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_correction_team_lead" model="ir.rule">
|
|
<field name="name">Correction: Team Lead sees direct reports</field>
|
|
<field name="model_id" ref="model_fusion_clock_correction"/>
|
|
<field name="domain_force">['|', ('employee_id.user_id', '=', user.id), ('employee_id.parent_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_team_lead'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_correction_manager" model="ir.rule">
|
|
<field name="name">Correction: Manager full access</field>
|
|
<field name="model_id" ref="model_fusion_clock_correction"/>
|
|
<field name="domain_force">[(1, '=', 1)]</field>
|
|
<field name="groups" eval="[(4, ref('group_fusion_clock_manager'))]"/>
|
|
</record>
|
|
|
|
<!-- ================================================================
|
|
Portal Access
|
|
================================================================ -->
|
|
<record id="rule_hr_attendance_portal" model="ir.rule">
|
|
<field name="name">HR Attendance: Portal user sees own</field>
|
|
<field name="model_id" ref="hr_attendance.model_hr_attendance"/>
|
|
<field name="domain_force">[('employee_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('base.group_portal'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_clock_location_portal" model="ir.rule">
|
|
<field name="name">Clock Location: Portal user sees active</field>
|
|
<field name="model_id" ref="model_fusion_clock_location"/>
|
|
<field name="domain_force">[('active', '=', True)]</field>
|
|
<field name="groups" eval="[(4, ref('base.group_portal'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_clock_report_portal" model="ir.rule">
|
|
<field name="name">Clock Report: Portal user sees own</field>
|
|
<field name="model_id" ref="model_fusion_clock_report"/>
|
|
<field name="domain_force">[('employee_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('base.group_portal'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_clock_penalty_portal" model="ir.rule">
|
|
<field name="name">Clock Penalty: Portal user sees own</field>
|
|
<field name="model_id" ref="model_fusion_clock_penalty"/>
|
|
<field name="domain_force">[('employee_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('base.group_portal'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_activity_log_portal" model="ir.rule">
|
|
<field name="name">Activity Log: Portal user sees own</field>
|
|
<field name="model_id" ref="model_fusion_clock_activity_log"/>
|
|
<field name="domain_force">[('employee_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('base.group_portal'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_leave_request_portal" model="ir.rule">
|
|
<field name="name">Leave Request: Portal user sees own</field>
|
|
<field name="model_id" ref="model_fusion_clock_leave_request"/>
|
|
<field name="domain_force">[('employee_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('base.group_portal'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_correction_portal" model="ir.rule">
|
|
<field name="name">Correction: Portal user sees own</field>
|
|
<field name="model_id" ref="model_fusion_clock_correction"/>
|
|
<field name="domain_force">[('employee_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('base.group_portal'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
<record id="rule_schedule_portal" model="ir.rule">
|
|
<field name="name">Schedule: Portal user sees own</field>
|
|
<field name="model_id" ref="model_fusion_clock_schedule"/>
|
|
<field name="domain_force">[('employee_id.user_id', '=', user.id)]</field>
|
|
<field name="groups" eval="[(4, ref('base.group_portal'))]"/>
|
|
<field name="perm_read" eval="True"/>
|
|
<field name="perm_write" eval="False"/>
|
|
<field name="perm_create" eval="False"/>
|
|
<field name="perm_unlink" eval="False"/>
|
|
</record>
|
|
|
|
</odoo>
|