1 Commits

119 changed files with 2487 additions and 1694 deletions

View File

@@ -1,20 +0,0 @@
# Configuration for known file extensions
[*.{css,js,json,less,md,py,rst,sass,scss,xml,yaml,yml}]
charset = utf-8
end_of_line = lf
indent_size = 4
indent_style = space
insert_final_newline = true
trim_trailing_whitespace = true
[*.{json,yml,yaml,rst,md}]
indent_size = 2
# Do not configure editor for libs and autogenerated content
[{*/static/{lib,src/lib}/**,*/static/description/index.html,*/readme/../README.rst}]
charset = unset
end_of_line = unset
indent_size = unset
indent_style = unset
insert_final_newline = false
trim_trailing_whitespace = false

View File

@@ -1,188 +0,0 @@
env:
browser: true
es6: true
# See https://github.com/OCA/odoo-community.org/issues/37#issuecomment-470686449
parserOptions:
ecmaVersion: 2019
overrides:
- files:
- "**/*.esm.js"
parserOptions:
sourceType: module
# Globals available in Odoo that shouldn't produce errorings
globals:
_: readonly
$: readonly
fuzzy: readonly
jQuery: readonly
moment: readonly
odoo: readonly
openerp: readonly
owl: readonly
luxon: readonly
# Styling is handled by Prettier, so we only need to enable AST rules;
# see https://github.com/OCA/maintainer-quality-tools/pull/618#issuecomment-558576890
rules:
accessor-pairs: warn
array-callback-return: warn
callback-return: warn
capitalized-comments:
- warn
- always
- ignoreConsecutiveComments: true
ignoreInlineComments: true
complexity:
- warn
- 15
constructor-super: warn
dot-notation: warn
eqeqeq: warn
global-require: warn
handle-callback-err: warn
id-blacklist: warn
id-match: warn
init-declarations: error
max-depth: warn
max-nested-callbacks: warn
max-statements-per-line: warn
no-alert: warn
no-array-constructor: warn
no-caller: warn
no-case-declarations: warn
no-class-assign: warn
no-cond-assign: error
no-const-assign: error
no-constant-condition: warn
no-control-regex: warn
no-debugger: error
no-delete-var: warn
no-div-regex: warn
no-dupe-args: error
no-dupe-class-members: error
no-dupe-keys: error
no-duplicate-case: error
no-duplicate-imports: error
no-else-return: warn
no-empty-character-class: warn
no-empty-function: error
no-empty-pattern: error
no-empty: warn
no-eq-null: error
no-eval: error
no-ex-assign: error
no-extend-native: warn
no-extra-bind: warn
no-extra-boolean-cast: warn
no-extra-label: warn
no-fallthrough: warn
no-func-assign: error
no-global-assign: error
no-implicit-coercion:
- warn
- allow: ["~"]
no-implicit-globals: warn
no-implied-eval: warn
no-inline-comments: warn
no-inner-declarations: warn
no-invalid-regexp: warn
no-irregular-whitespace: warn
no-iterator: warn
no-label-var: warn
no-labels: warn
no-lone-blocks: warn
no-lonely-if: error
no-mixed-requires: error
no-multi-str: warn
no-native-reassign: error
no-negated-condition: warn
no-negated-in-lhs: error
no-new-func: warn
no-new-object: warn
no-new-require: warn
no-new-symbol: warn
no-new-wrappers: warn
no-new: warn
no-obj-calls: warn
no-octal-escape: warn
no-octal: warn
no-param-reassign: warn
no-path-concat: warn
no-process-env: warn
no-process-exit: warn
no-proto: warn
no-prototype-builtins: warn
no-redeclare: warn
no-regex-spaces: warn
no-restricted-globals: warn
no-restricted-imports: warn
no-restricted-modules: warn
no-restricted-syntax: warn
no-return-assign: error
no-script-url: warn
no-self-assign: warn
no-self-compare: warn
no-sequences: warn
no-shadow-restricted-names: warn
no-shadow: warn
no-sparse-arrays: warn
no-sync: warn
no-this-before-super: warn
no-throw-literal: warn
no-undef-init: warn
no-undef: error
no-unmodified-loop-condition: warn
no-unneeded-ternary: error
no-unreachable: error
no-unsafe-finally: error
no-unused-expressions: error
no-unused-labels: error
no-unused-vars: error
no-use-before-define: error
no-useless-call: warn
no-useless-computed-key: warn
no-useless-concat: warn
no-useless-constructor: warn
no-useless-escape: warn
no-useless-rename: warn
no-void: warn
no-with: warn
operator-assignment: [error, always]
prefer-const: warn
radix: warn
require-yield: warn
sort-imports: warn
spaced-comment: [error, always]
strict: [error, function]
use-isnan: error
valid-jsdoc:
- warn
- prefer:
arg: param
argument: param
augments: extends
constructor: class
exception: throws
func: function
method: function
prop: property
return: returns
virtual: abstract
yield: yields
preferType:
array: Array
bool: Boolean
boolean: Boolean
number: Number
object: Object
str: String
string: String
requireParamDescription: false
requireReturn: false
requireReturnDescription: false
requireReturnType: false
valid-typeof: warn
yoda: warn

View File

@@ -1,42 +0,0 @@
name: pre-commit
on:
pull_request:
branches:
- "16.0*"
jobs:
pre-commit:
runs-on: ubuntu-22.04
steps:
- uses: actions/checkout@v4
- uses: actions/setup-python@v5
with:
python-version: "3.11"
- name: Get python version
run: echo "PY=$(python -VV | sha256sum | cut -d' ' -f1)" >> $GITHUB_ENV
# - uses: actions/cache@v4
# with:
# path: ~/.cache/pre-commit
# key: pre-commit|${{ env.PY }}|${{ hashFiles('.pre-commit-config.yaml') }}
- name: Install pre-commit
run: pip install pre-commit
- name: Run pre-commit
run: pre-commit run --all-files --show-diff-on-failure --color=always
env:
# Consider valid a PR that changes README fragments but doesn't
# change the README.rst file itself. It's not really a problem
# because the bot will update it anyway after merge. This way, we
# lower the barrier for functional contributors that want to fix the
# readme fragments, while still letting developers get README
# auto-generated (which also helps functionals when using runboat).
# DOCS https://pre-commit.com/#temporarily-disabling-hooks
SKIP: oca-gen-addon-readme
- name: Check that all files generated by pre-commit are in git
run: |
newfiles="$(git ls-files --others --exclude-from=.gitignore)"
if [ "$newfiles" != "" ] ; then
echo "Please check-in the following files:"
echo "$newfiles"
exit 1
fi

View File

@@ -1,150 +0,0 @@
exclude: |
(?x)
# NOT INSTALLABLE ADDONS
# END NOT INSTALLABLE ADDONS
# Files and folders generated by bots, to avoid loops
^setup/|/static/description/index\.html$|
# We don't want to mess with tool-generated files
.svg$|/tests/([^/]+/)?cassettes/|^.copier-answers.yml$|^.github/|^eslint.config.cjs|^prettier.config.cjs|
# Maybe reactivate this when all README files include prettier ignore tags?
^README\.md$|
# Library files can have extraneous formatting (even minimized)
/static/(src/)?lib/|
# Repos using Sphinx to generate docs don't need prettying
^docs/_templates/.*\.html$|
# Don't bother non-technical authors with formatting issues in docs
readme/.*\.(rst|md)$|
# Ignore build and dist directories in addons
/build/|/dist/|
# Ignore test files in addons
/tests/samples/.*|
# You don't usually want a bot to modify your legal texts
(LICENSE.*|COPYING.*)
default_language_version:
python: python3
node: "16.17.0"
repos:
- repo: local
hooks:
# These files are most likely copier diff rejection junks; if found,
# review them manually, fix the problem (if needed) and remove them
- id: forbidden-files
name: forbidden files
entry: found forbidden files; remove them
language: fail
files: "\\.rej$"
- id: en-po-files
name: en.po files cannot exist
entry: found a en.po file
language: fail
files: '[a-zA-Z0-9_]*/i18n/en\.po$'
- repo: https://github.com/oca/maintainer-tools
rev: f9b919b9868143135a9c9cb03021089cabba8223
hooks:
# update the NOT INSTALLABLE ADDONS section above
- id: oca-update-pre-commit-excluded-addons
- id: oca-fix-manifest-website
entry:
bash -c 'oca-fix-manifest-website "https://git.elabore.coop/elabore/$(basename
$(git rev-parse --show-toplevel))"'
- id: oca-gen-addon-readme
entry:
bash -c 'oca-gen-addon-readme
--addons-dir=.
--branch=$(git symbolic-ref
refs/remotes/origin/HEAD | sed "s@^refs/remotes/origin/@@")
--repo-name=$(basename $(git rev-parse --show-toplevel))
--org-name="Elabore"
--if-source-changed --keep-source-digest'
- repo: https://github.com/OCA/odoo-pre-commit-hooks
rev: v0.1.4
hooks:
- id: oca-checks-odoo-module
- id: oca-checks-po
args:
- --disable=po-pretty-format
- repo: local
hooks:
- id: prettier
name: prettier (with plugin-xml)
entry: prettier
args:
- --write
- --list-different
- --ignore-unknown
types: [text]
files: \.(css|htm|html|js|json|jsx|less|md|scss|toml|ts|xml|yaml|yml)$
language: node
additional_dependencies:
- "prettier@2.7.1"
- "@prettier/plugin-xml@2.2.0"
- repo: local
hooks:
- id: eslint
name: eslint
entry: eslint
args:
- --color
- --fix
verbose: true
types: [javascript]
language: node
additional_dependencies:
- "eslint@8.24.0"
- "eslint-plugin-jsdoc@"
- repo: https://github.com/pre-commit/pre-commit-hooks
rev: v5.0.0
hooks:
- id: trailing-whitespace
# exclude autogenerated files
exclude: /README\.rst$|\.pot?$
- id: end-of-file-fixer
# exclude autogenerated files
exclude: /README\.rst$|\.pot?$
- id: debug-statements
- id: fix-encoding-pragma
args: ["--remove"]
- id: check-case-conflict
- id: check-docstring-first
- id: check-executables-have-shebangs
- id: check-merge-conflict
# exclude files where underlines are not distinguishable from merge conflicts
exclude: /README\.rst$|^docs/.*\.rst$
- id: check-symlinks
- id: check-xml
- id: mixed-line-ending
args: ["--fix=lf"]
- repo: https://github.com/PyCQA/docformatter
rev: v1.7.7
hooks:
- id: docformatter
args: [
"--in-place", # modify the files
"--recursive", # run on all the files
"--wrap-summaries",
"88", # max length of 1st line
"--wrap-descriptions",
"88", # max length of other lines
"--pre-summary-newline", # new line before a long summary
"--make-summary-multi-line", # force summary on multilines
]
additional_dependencies: ["tomli"] # if Python <3.11
- repo: https://github.com/astral-sh/ruff-pre-commit
rev: v0.12.0
hooks:
- id: ruff
args: [--fix, --exit-non-zero-on-fix]
- id: ruff-format
- repo: https://github.com/OCA/pylint-odoo
rev: v9.1.3
hooks:
- id: pylint_odoo
name: pylint with optional checks
args:
- --rcfile=.pylintrc
- --exit-zero
verbose: true
- id: pylint_odoo
args:
- --rcfile=.pylintrc-mandatory

View File

@@ -1,8 +0,0 @@
# Defaults for all prettier-supported languages.
# Prettier will complete this with settings from .editorconfig file.
bracketSpacing: false
printWidth: 88
proseWrap: always
semi: true
trailingComma: "es5"
xmlWhitespaceSensitivity: "strict"

123
.pylintrc
View File

@@ -1,123 +0,0 @@
[MASTER]
load-plugins=pylint_odoo
score=n
[ODOOLINT]
readme-template-url="https://github.com/OCA/maintainer-tools/blob/master/template/module/README.rst"
manifest-required-authors=Elabore
manifest-required-keys=license
manifest-deprecated-keys=description,active
license-allowed=AGPL-3,GPL-2,GPL-2 or any later version,GPL-3,GPL-3 or any later version,LGPL-3
valid-odoo-versions=16.0
[MESSAGES CONTROL]
disable=all
# This .pylintrc contains optional AND mandatory checks and is meant to be
# loaded in an IDE to have it check everything, in the hope this will make
# optional checks more visible to contributors who otherwise never look at a
# green travis to see optional checks that failed.
# .pylintrc-mandatory containing only mandatory checks is used the pre-commit
# config as a blocking check.
enable=anomalous-backslash-in-string,
api-one-deprecated,
api-one-multi-together,
assignment-from-none,
attribute-deprecated,
class-camelcase,
dangerous-default-value,
dangerous-view-replace-wo-priority,
development-status-allowed,
duplicate-id-csv,
duplicate-key,
duplicate-xml-fields,
duplicate-xml-record-id,
eval-referenced,
eval-used,
incoherent-interpreter-exec-perm,
license-allowed,
manifest-author-string,
manifest-deprecated-key,
manifest-required-author,
manifest-required-key,
manifest-version-format,
method-compute,
method-inverse,
method-required-super,
method-search,
openerp-exception-warning,
pointless-statement,
pointless-string-statement,
print-used,
redundant-keyword-arg,
redundant-modulename-xml,
reimported,
relative-import,
return-in-init,
rst-syntax-error,
sql-injection,
too-few-format-args,
translation-field,
translation-required,
unreachable,
use-vim-comment,
wrong-tabs-instead-of-spaces,
xml-syntax-error,
attribute-string-redundant,
character-not-valid-in-resource-link,
consider-merging-classes-inherited,
context-overridden,
create-user-wo-reset-password,
dangerous-filter-wo-user,
dangerous-qweb-replace-wo-priority,
deprecated-data-xml-node,
deprecated-openerp-xml-node,
duplicate-po-message-definition,
except-pass,
file-not-used,
invalid-commit,
manifest-maintainers-list,
missing-newline-extrafiles,
missing-readme,
missing-return,
odoo-addons-relative-import,
old-api7-method-defined,
po-msgstr-variables,
po-syntax-error,
renamed-field-parameter,
resource-not-exist,
str-format-used,
test-folder-imported,
translation-contains-variable,
translation-positional-used,
unnecessary-utf8-coding-comment,
website-manifest-key-not-valid-uri,
xml-attribute-translatable,
xml-deprecated-qweb-directive,
xml-deprecated-tree-attribute,
external-request-timeout,
# messages that do not cause the lint step to fail
consider-merging-classes-inherited,
create-user-wo-reset-password,
dangerous-filter-wo-user,
deprecated-module,
file-not-used,
invalid-commit,
missing-manifest-dependency,
missing-newline-extrafiles,
missing-readme,
no-utf8-coding-comment,
odoo-addons-relative-import,
old-api7-method-defined,
redefined-builtin,
too-complex,
unnecessary-utf8-coding-comment
[REPORTS]
msg-template={path}:{line}: [{msg_id}({symbol}), {obj}] {msg}
output-format=colorized
reports=no

View File

@@ -1,98 +0,0 @@
[MASTER]
load-plugins=pylint_odoo
score=n
[ODOOLINT]
readme-template-url="https://github.com/OCA/maintainer-tools/blob/master/template/module/README.rst"
manifest-required-authors=Elabore
manifest-required-keys=license
manifest-deprecated-keys=description,active
license-allowed=AGPL-3,GPL-2,GPL-2 or any later version,GPL-3,GPL-3 or any later version,LGPL-3
valid-odoo-versions=16.0
[MESSAGES CONTROL]
disable=all
enable=anomalous-backslash-in-string,
api-one-deprecated,
api-one-multi-together,
assignment-from-none,
attribute-deprecated,
class-camelcase,
dangerous-default-value,
dangerous-view-replace-wo-priority,
development-status-allowed,
duplicate-id-csv,
duplicate-key,
duplicate-xml-fields,
duplicate-xml-record-id,
eval-referenced,
eval-used,
incoherent-interpreter-exec-perm,
license-allowed,
manifest-author-string,
manifest-deprecated-key,
manifest-required-author,
manifest-required-key,
manifest-version-format,
method-compute,
method-inverse,
method-required-super,
method-search,
openerp-exception-warning,
pointless-statement,
pointless-string-statement,
print-used,
redundant-keyword-arg,
redundant-modulename-xml,
reimported,
relative-import,
return-in-init,
rst-syntax-error,
sql-injection,
too-few-format-args,
translation-field,
translation-required,
unreachable,
use-vim-comment,
wrong-tabs-instead-of-spaces,
xml-syntax-error,
attribute-string-redundant,
character-not-valid-in-resource-link,
consider-merging-classes-inherited,
context-overridden,
create-user-wo-reset-password,
dangerous-filter-wo-user,
dangerous-qweb-replace-wo-priority,
deprecated-data-xml-node,
deprecated-openerp-xml-node,
duplicate-po-message-definition,
except-pass,
file-not-used,
invalid-commit,
manifest-maintainers-list,
missing-newline-extrafiles,
missing-readme,
missing-return,
odoo-addons-relative-import,
old-api7-method-defined,
po-msgstr-variables,
po-syntax-error,
renamed-field-parameter,
resource-not-exist,
str-format-used,
test-folder-imported,
translation-contains-variable,
translation-positional-used,
unnecessary-utf8-coding-comment,
website-manifest-key-not-valid-uri,
xml-attribute-translatable,
xml-deprecated-qweb-directive,
xml-deprecated-tree-attribute,
external-request-timeout
[REPORTS]
msg-template={path}:{line}: [{msg_id}({symbol}), {obj}] {msg}
output-format=colorized
reports=no

View File

@@ -1,31 +0,0 @@
target-version = "py310"
fix = true
[lint]
extend-select = [
"B",
"C90",
"E501", # line too long (default 88)
"I", # isort
"UP", # pyupgrade
]
extend-safe-fixes = ["UP008"]
exclude = ["setup/*"]
[format]
exclude = ["setup/*"]
[lint.per-file-ignores]
"__init__.py" = ["F401", "I001"] # ignore unused and unsorted imports in __init__.py
"__manifest__.py" = ["B018"] # useless expression
[lint.isort]
section-order = ["future", "standard-library", "third-party", "odoo", "odoo-addons", "first-party", "local-folder"]
[lint.isort.sections]
"odoo" = ["odoo"]
"odoo-addons" = ["odoo.addons"]
[lint.mccabe]
max-complexity = 16

View File

@@ -0,0 +1,45 @@
=================
project_assignees
=================
Add multiple assignees field to project task
Installation
============
Use Odoo normal procedure to install add-ons to install
``project_assignees``.
Known issues / Roadmap
======================
Bug Tracker
===========
Bugs are tracked on `our issues website
<https://github.com/elabore-coop/project-tools/issues>`_. In case of
trouble, please check there if your issue has already been
reported. If you spotted it first, help us smashing it by providing a
detailed and welcomed feedback.
Credits
=======
Images
------
* Elabore: `Icon <https://elabore.coop/web/image/res.company/1/logo?unique=f3db262>`_.
Contributors
------------
* Stéphan Sainléger <https://github.com/stephansainleger>
Funders
-------
The development of this module has been financially supported by:
* Elabore (https://elabore.coop)
Maintainer
----------
This module is maintained by Elabore.

View File

@@ -0,0 +1,3 @@
# -*- coding: utf-8 -*-
from . import models

View File

@@ -0,0 +1,94 @@
# Copyright 2022 Stéphan Sainléger (Elabore)
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
{
"name": "project_assignees",
"version": "16.0.1.1.0",
"author": "Elabore",
"website": "https://github.com/elabore-coop/project-tools",
"maintainer": "Stéphan Sainléger",
"license": "AGPL-3",
"category": "Tools",
"summary": "Add multiple assignees field to project task",
"description": """
:image: https://img.shields.io/badge/licence-AGPL--3-blue.svg
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
:alt: License: AGPL-3
=================
project_assignees
=================
Add multiple assignees field to project task.
Installation
============
Install ``project_assignees``, all dependencies will be installed by default.
Known issues / Roadmap
======================
None yet.
Bug Tracker
===========
Bugs are tracked on `our issues website
<https://github.com/elabore-coop/project-tools/issues>`_. In case of
trouble, please check there if your issue has already been
reported. If you spotted it first, help us smashing it by providing a
detailed and welcomed feedback.
Credits
=======
Images
------
* Elabore: `Icon <https://elabore.coop/web/image/res.company/1/logo?unique=f3db262>`_.
Contributors
------------
* Stéphan Sainléger <https://github.com/stephansainleger>
* Valentin Lab <valentin.lab@kalysto.org>
Funders
-------
The development of this module has been financially supported by:
* Elabore (https://elabore.coop)
Maintainer
----------
This module is maintained by Elabore.
""",
# any module necessary for this one to work correctly
"depends": [
"base",
"project",
"project_task_portal_form",
],
"qweb": [
# "static/src/xml/*.xml",
],
"external_dependencies": {
"python": [],
},
# always loaded
"data": [
"views/project_task.xml",
"views/portal_template.xml",
],
# only loaded in demonstration mode
"demo": [],
"js": [],
"css": [],
"installable": True,
# Install this module automatically if all dependency have been previously
# and independently installed. Used for synergetic or glue modules.
"auto_install": False,
"application": False,
}

View File

@@ -0,0 +1 @@
This directory should contain the *.po for Odoo translation.

View File

@@ -0,0 +1,36 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * project_assignees
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 16.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2023-11-02 07:56+0000\n"
"PO-Revision-Date: 2023-11-02 07:56+0000\n"
"Last-Translator: \n"
"Language-Team: \n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: \n"
#. module: project_assignees
#: model_terms:ir.ui.view,arch_db:project_assignees.portal_my_task_assignees
msgid "<strong>Other assignees</strong>"
msgstr "<strong>Autres intervenants</strong>"
#. module: project_assignees
#: model:ir.model.fields,field_description:project_assignees.field_project_task__assignee_ids
msgid "Assignees"
msgstr "Autres assignations"
#. module: project_assignees
#: model_terms:ir.ui.view,arch_db:project_assignees.portal_my_task_assignees
msgid "Contact"
msgstr ""
#. module: project_assignees
#: model:ir.model,name:project_assignees.model_project_task
msgid "Task"
msgstr "Tâche"

View File

@@ -0,0 +1,2 @@
from . import project_task

View File

@@ -0,0 +1,9 @@
from odoo import models, fields
class Task(models.Model):
_inherit = "project.task"
assignee_ids = fields.Many2many('res.users', 'assignee_ids_rel', string='Assignees')

View File

@@ -0,0 +1,24 @@
<?xml version="1.0" encoding="utf-8"?>
<odoo>
<template id="portal_my_task_assignees" name="My Task: Assignees" inherit_id="project.portal_my_task" priority="40">
<xpath expr="//t[@t-foreach='task.user_ids']" position="after">
<div class="col-12 col-md-12 pb-2" t-if="task.assignee_ids">
<strong>Other assignees</strong>
<div class="row">
<t t-foreach="task.assignee_ids" t-as="user">
<div class="d-flex mb-3 flex-nowrap">
<img class="rounded-circle mt-1 o_portal_contact_img" t-att-src="image_data_uri(user.avatar_1024)" alt="Contact"/>
<div class="ms-2">
<div t-esc="user" t-options='{"widget": "contact", "fields": ["name"]}'/>
<a t-attf-href="tel:{{user.phone}}" t-if="user.phone"><div t-esc="user" t-options='{"widget": "contact", "fields": ["phone"]}'/></a>
<a t-if="user.email" class="text-break" t-attf-href="mailto:{{user.email}}">
<div t-out="user" t-options='{"widget": "contact", "fields": ["email"]}'/>
</a>
</div>
</div>
</t>
</div>
</div>
</xpath>
</template>
</odoo>

View File

@@ -0,0 +1,29 @@
<?xml version="1.0" encoding="UTF-8"?>
<odoo>
<record id="view_task_form2_assignees" model="ir.ui.view">
<field name="name">project.task.form.assignees</field>
<field name="model">project.task</field>
<field name="inherit_id" ref="project.view_task_form2" />
<field name="priority" eval="99" />
<field name="arch" type="xml">
<xpath expr="//field[@name='user_ids']" position="after">
<field name="assignee_ids" widget="many2many_tags" />
</xpath>
</field>
</record>
<record id="view_task_search_form_assignees" model="ir.ui.view">
<field name="name">project.task.search.form.assignees</field>
<field name="model">project.task</field>
<field name="inherit_id" ref="project.view_task_search_form" />
<field name="priority" eval="99" />
<field name="arch" type="xml">
<filter name="my_tasks" position="attributes">
<attribute name="domain">['|', ('user_ids', 'in', uid), ('assignee_ids', 'in', uid)]</attribute>
</filter>
<xpath expr="//field[@name='partner_id']" position="after">
<field name="assignee_ids" filter_domain="[('assignee_ids.user_ids.name', 'ilike', self)]"/>
</xpath>
</field>
</record>
</odoo>

View File

@@ -1 +1,3 @@
# -*- coding: utf-8 -*-
from . import models, controllers

View File

@@ -3,15 +3,73 @@
{
"name": "project_average_acceptable_time",
"version": "18.0.1.0.0",
"version": "16.0.1.0.0",
"author": "Elabore",
"website": "https://git.elabore.coop/elabore/project-tools",
"website": "https://github.com/elabore-coop/project-tools",
"maintainer": "Clément Thomas",
"license": "AGPL-3",
"category": "Tools",
"summary": "Task validation without customer agreement",
"description": """
:image: https://img.shields.io/badge/licence-AGPL--3-blue.svg
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
:alt: License: AGPL-3
=================
project_average_acceptable_time
=================
Installation
============
Install ``project_average_acceptable_time``, all dependencies will be installed by default.
Known issues / Roadmap
======================
None yet.
Bug Tracker
===========
Bugs are tracked on `our issues website
<https://github.com/elabore-coop/project-tools/issues>`_. In case of
trouble, please check there if your issue has already been
reported. If you spotted it first, help us smashing it by providing a
detailed and welcomed feedback.
Credits
=======
Images
------
* Elabore: `Icon <https://elabore.coop/web/image/res.company/1/logo?unique=f3db262>`_.
Contributors
------------
* Clément Thomas
Funders
-------
The development of this module has been financially supported by:
* Elabore (https://elabore.coop)
Maintainer
----------
This module is maintained by Elabore.
""",
# any module necessary for this one to work correctly
"depends": ["base", "project", "project_user_default_project"],
"depends": [
"base",
"project",
"project_user_default_project"
],
"qweb": [
# "static/src/xml/*.xml",
],
@@ -19,7 +77,10 @@
"python": [],
},
# always loaded
"data": ["views/project_project.xml", "views/portal_home_template.xml"],
"data": [
"views/project_project.xml",
"views/portal_home_template.xml"
],
# only loaded in demonstration mode
"demo": [],
"js": [],

View File

@@ -1 +1,2 @@
from . import custom_portal
from . import custom_portal

View File

@@ -1,26 +1,21 @@
# Copyright 2020 Lokavaluto ()
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from odoo.http import request, route
from odoo.addons.portal.controllers.portal import CustomerPortal
class CustomCustomerPortal(CustomerPortal):
@route(["/my/account"], type="http", auth="user", website=True)
def account(self, redirect=None, **post):
self.OPTIONAL_BILLING_FIELDS.append(
"average_acceptable_time"
) # unecessary save in res partner, but necessary to avoid error on form post
response = super().account(redirect, **post)
self.OPTIONAL_BILLING_FIELDS.append("average_acceptable_time") #unecessary save in res partner, but necessary to avoid error on form post
response = super(CustomCustomerPortal, self).account(redirect, **post)
if post and request.httprequest.method == "POST":
error, error_message = self.details_form_validate(post)
if not error:
user = request.env.user
if user.default_project_id and post["average_acceptable_time"]:
user.default_project_id.average_acceptable_time = post[
"average_acceptable_time"
]
user.default_project_id.average_acceptable_time = post["average_acceptable_time"]
return response

View File

@@ -1,2 +1,3 @@
from . import project_project
from . import res_partner
from . import res_partner

View File

@@ -1,7 +1,10 @@
from odoo import fields, models
from odoo import models, fields, _, api
class Project(models.Model):
_inherit = "project.project"
average_acceptable_time = fields.Float("Average acceptable time")
average_acceptable_time = fields.Float('Average acceptable time')

View File

@@ -1,9 +1,10 @@
from odoo import fields, models
from odoo import models, fields, _, api
class ResPartner(models.Model):
_inherit = "res.partner"
average_acceptable_time = fields.Float(
"Average acceptable time"
) # not used, but necessary to post custom field from /my/account
average_acceptable_time = fields.Float('Average acceptable time') # not used, but necessary to post custom field from /my/account

View File

@@ -1,43 +1,26 @@
<?xml version="1.0" encoding="utf-8" ?>
<?xml version="1.0" encoding="utf-8"?>
<odoo>
<!-- Template /my/home -->
<template
id="portal_my_home_average_acceptable_time"
name="Portal My Home: Average acceptable time"
inherit_id="portal.side_content"
priority="40"
>
<xpath expr="//div[@class='o_portal_my_details']" position="inside">
<hr class="mt-1 mb-0" />
<b>Average acceptable time</b>: <t
t-esc="user_id.default_project_id.average_acceptable_time"
t-options="{'widget': 'float_time'}"
/>
<template id="portal_my_home_average_acceptable_time" name="Portal My Home: Average acceptable time" inherit_id="portal.portal_layout" priority="40">
<xpath expr="//div[hasclass('o_portal_my_details')]" position="inside">
<hr class="mt-1 mb-0"/>
<b>Average acceptable time</b>: <t t-esc="user_id.default_project_id.average_acceptable_time" t-options="{'widget': 'float_time'}"/>
</xpath>
</template>
<!-- Template /my/account -->
<template
id="portal_my_details_average_acceptable_time"
name="Portal My details: Average acceptable time"
inherit_id="portal.portal_my_details_fields"
>
<xpath expr="//input[@name='csrf_token']/.." position="inside">
<div
t-attf-class="form-group #{error.get('average_acceptable_time') and 'o_has_error' or ''} col-xl-6"
>
<label
class="col-form-label"
for="average_acceptable_time"
>Average acceptable time (h)</label>
<input
type="text"
name="average_acceptable_time"
t-attf-class="form-control #{error.get('average_acceptable_time') and 'is-invalid' or ''}"
t-att-value="average_acceptable_time or user_id.default_project_id.average_acceptable_time"
<template id="portal_my_details_average_acceptable_time" name="Portal My details: Average acceptable time" inherit_id="portal.portal_my_details">
<xpath expr="//input[@name='redirect']" position="before">
<div t-attf-class="form-group #{error.get('average_acceptable_time') and 'o_has_error' or ''} col-xl-6">
<label class="col-form-label" for="average_acceptable_time">Average acceptable time (h)</label>
<input
type="text"
name="average_acceptable_time"
t-attf-class="form-control #{error.get('average_acceptable_time') and 'is-invalid' or ''}"
t-att-value="average_acceptable_time or user_id.default_project_id.average_acceptable_time"
t-options="{'widget': 'float_time'}"
/>
/>
</div>
</xpath>
</template>
</odoo>
</odoo>

View File

@@ -1,47 +1,36 @@
<?xml version="1.0" encoding="UTF-8" ?>
<?xml version="1.0" encoding="UTF-8"?>
<odoo>
<!--Project KANBAN -->
<record
id="project_project_view_kanban_inherit_average_acceptable_time"
model="ir.ui.view"
>
<field
name="name"
>project.project.kanban.inherit.average.acceptable.time</field>
<record id="project_project_view_kanban_inherit_average_acceptable_time" model="ir.ui.view">
<field name="name">project.project.kanban.inherit.average.acceptable.time</field>
<field name="model">project.project</field>
<field name="inherit_id" ref="project.view_project_kanban" />
<field name="inherit_id" ref="project.view_project_kanban"/>
<field name="priority">999</field>
<field name="arch" type="xml">
<xpath expr="//div[hasclass('o_project_kanban_main')]" position="inside">
<xpath expr="//div[@class='o_project_kanban_main ']" position="inside">
<div class="o_project_kanban_boxes">
<span>Average acceptable time:<![CDATA[&nbsp;]]></span>
<field name="average_acceptable_time" widget="float_time" />
<field name="average_acceptable_time" widget="float_time"/>
</div>
</xpath>
</field>
</record>
<!-- Project FORM -->
<record
id="project_project_form_inherit_average_acceptable_time"
model="ir.ui.view"
>
<record id="project_project_form_inherit_average_acceptable_time" model="ir.ui.view">
<field name="name">project.project.form.inherit.average.acceptable.time</field>
<field name="model">project.project</field>
<field name="inherit_id" ref="project.edit_project" />
<field name="inherit_id" ref="project.edit_project"/>
<field name="arch" type="xml">
<xpath expr="//group[@name='extra_settings']" position="inside">
<div
class="o_settings_average_acceptable_time"
style="margin-top:10px;"
>
<xpath expr="//div[@id='subtask_settings']/div[@class='o_setting_right_pane']" position="inside">
<div class="o_settings_average_acceptable_time" style="margin-top:10px;">
<!-- <span>Average acceptable time:<![CDATA[&nbsp;]]></span> -->
<label for="average_acceptable_time" />
<field name="average_acceptable_time" widget="float_time" />
<label for="average_acceptable_time"/>
<field name="average_acceptable_time" widget="float_time"/>
</div>
</xpath>
</field>
</record>
</odoo>
</odoo>

View File

@@ -0,0 +1,46 @@
:image: https://img.shields.io/badge/licence-AGPL--3-blue.svg
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
:alt: License: AGPL-3
===============
project_funders
===============
Add list of funder and corresponding amount to project task.
Installation
============
Use Odoo normal module installation procedure to install
``project_funders``.
Known issues / Roadmap
======================
None yet.
Bug Tracker
===========
Bugs are tracked on `our issues website <https://github.com/elabore-coop/project-tools/issues>`_. In case of
trouble, please check there if your issue has already been
reported. If you spotted it first, help us smashing it by providing a
detailed and welcomed feedback.
Credits
=======
Contributors
------------
* Nicolas Jeudy <https://github.com/njeudy>
* Valentin Lab
Funders
-------
The development of this module has been financially supported by:
* Alusage (https://alusage.fr)
Maintainer
----------
This module is maintained by Alusage and Elabore.

View File

@@ -0,0 +1 @@
from . import models

View File

@@ -0,0 +1,87 @@
# Copyright 2022 Nicolas Jeudy (Alusage)
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
{
"name": "project_funders",
"version": "16.0.1.0.0",
"author": "Alusage",
"website": "https://alusage.fr",
"data": [
# "security/security.xml",
# "security/ir.model.access.csv",
# "views/menu.xml",
# "data/data.xml",
],
"author": "Alusage, Elabore",
"maintainer": "Nicolas Jeudy",
"license": "AGPL-3",
"category": "Tools",
"summary": " Odoo module.",
"description": """
:image: https://img.shields.io/badge/licence-AGPL--3-blue.svg
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
:alt: License: AGPL-3
===============
project_funders
===============
Add list of funder and corresponding amount to project task.
Installation
============
Use Odoo normal module installation procedure to install
``project_funders``.
Known issues / Roadmap
======================
None yet.
Bug Tracker
===========
Bugs are tracked on `our issues website <https://github.com/elabore-coop/project-tools/issues>`_. In case of
trouble, please check there if your issue has already been
reported. If you spotted it first, help us smashing it by providing a
detailed and welcomed feedback.
Credits
=======
Contributors
------------
* Nicolas Jeudy <https://github.com/njeudy>
* Valentin Lab
Funders
-------
The development of this module has been financially supported by:
* Alusage (https://alusage.fr)
Maintainer
----------
This module is maintained by Alusage and Elabore.
""",
# any module necessary for this one to work correctly
"depends": ["base", "project"],
"qweb": [
# "static/src/xml/*.xml",
],
"external_dependencies": {
"python": [],
},
# always loaded
"data": ["security/ir.model.access.csv", "views/project_task_view.xml"],
# only loaded in demonstration mode
"demo": [],
"js": [],
"css": [],
"installable": True,
# Install this module automatically if all dependency have been previously
# and independently installed. Used for synergetic or glue modules.
"auto_install": False,
"application": False,
}

View File

@@ -0,0 +1 @@
This directory should contain the *.po for Odoo translation.

105
project_funders/i18n/fr.po Normal file
View File

@@ -0,0 +1,105 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * project_funders
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 14.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2022-07-26 06:53+0000\n"
"PO-Revision-Date: 2022-07-26 06:53+0000\n"
"Last-Translator: \n"
"Language-Team: \n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: \n"
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_task__amount_total
msgid "Amount Total"
msgstr "Montant total"
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__create_uid
msgid "Created by"
msgstr "Créé par"
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__create_date
msgid "Created on"
msgstr "Créé le"
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__display_name
#: model:ir.model.fields,field_description:project_funders.field_project_task__display_name
msgid "Display Name"
msgstr "Nom affiché"
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__partner_id
msgid "Funder"
msgstr "Financeur"
#. module: project_funders
#: model:ir.model,name:project_funders.model_project_funder
msgid "Funder and amount for tasks"
msgstr "Financeur et montant sur les taches"
#. module: project_funders
#: model_terms:ir.ui.view,arch_db:project_funders.view_task_form2_inherit_project
msgid "Funder(s)"
msgstr "Financeur(s)"
#. module: project_funders
#: model_terms:ir.ui.view,arch_db:project_funders.view_task_kanban_inherit_project_funders
msgid "Funds:"
msgstr "Fonds&nbsp;:"
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__id
#: model:ir.model.fields,field_description:project_funders.field_project_task__id
msgid "ID"
msgstr "ID"
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder____last_update
#: model:ir.model.fields,field_description:project_funders.field_project_task____last_update
msgid "Last Modified on"
msgstr "Dernière modification le"
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__write_uid
msgid "Last Updated by"
msgstr "Derniere modifiation par"
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__write_date
msgid "Last Updated on"
msgstr "Dernière mise à jour le"
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__name
msgid "Name"
msgstr "Nom"
#. module: project_funders
#: model:ir.model,name:project_funders.model_project_task
#: model:ir.model.fields,field_description:project_funders.field_project_funder__task_id
msgid "Task"
msgstr "Tâche"
#. module: project_funders
#: model_terms:ir.ui.view,arch_db:project_funders.view_task_form2_inherit_project
msgid "Total"
msgstr "Total"
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__amount
msgid "Untaxed Amount"
msgstr "Montant HT"
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_task__funder_ids
msgid "funder"
msgstr "Financeur"

View File

@@ -0,0 +1,105 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * project_funders
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 14.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2022-07-26 07:41+0000\n"
"PO-Revision-Date: 2022-07-26 07:41+0000\n"
"Last-Translator: \n"
"Language-Team: \n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: \n"
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_task__amount_total
msgid "Amount Total"
msgstr ""
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__create_uid
msgid "Created by"
msgstr ""
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__create_date
msgid "Created on"
msgstr ""
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__display_name
#: model:ir.model.fields,field_description:project_funders.field_project_task__display_name
msgid "Display Name"
msgstr ""
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__partner_id
msgid "Funder"
msgstr ""
#. module: project_funders
#: model:ir.model,name:project_funders.model_project_funder
msgid "Funder and amount for tasks"
msgstr ""
#. module: project_funders
#: model_terms:ir.ui.view,arch_db:project_funders.view_task_form2_inherit_project
msgid "Funder(s)"
msgstr ""
#. module: project_funders
#: model_terms:ir.ui.view,arch_db:project_funders.view_task_kanban_inherit_project_funders
msgid "Funds:"
msgstr ""
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__id
#: model:ir.model.fields,field_description:project_funders.field_project_task__id
msgid "ID"
msgstr ""
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder____last_update
#: model:ir.model.fields,field_description:project_funders.field_project_task____last_update
msgid "Last Modified on"
msgstr ""
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__write_uid
msgid "Last Updated by"
msgstr ""
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__write_date
msgid "Last Updated on"
msgstr ""
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__name
msgid "Name"
msgstr ""
#. module: project_funders
#: model:ir.model,name:project_funders.model_project_task
#: model:ir.model.fields,field_description:project_funders.field_project_funder__task_id
msgid "Task"
msgstr ""
#. module: project_funders
#: model_terms:ir.ui.view,arch_db:project_funders.view_task_form2_inherit_project
msgid "Total"
msgstr ""
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_funder__amount
msgid "Untaxed Amount"
msgstr ""
#. module: project_funders
#: model:ir.model.fields,field_description:project_funders.field_project_task__funder_ids
msgid "funder"
msgstr ""

View File

@@ -0,0 +1,2 @@
from . import project_funders
from . import project_task

View File

@@ -0,0 +1,16 @@
from odoo import _, api, fields, models
class ProjectFunders(models.Model):
_name = "project.funder"
_description = "Funder and amount for tasks"
name = fields.Char(compute="_compute_name", string="Name")
partner_id = fields.Many2one("res.partner", string="Funder", required=True)
amount = fields.Float("Untaxed Amount", required=True)
task_id = fields.Many2one("project.task", string="Task")
@api.depends("partner_id", "amount")
def _compute_name(self):
for record in self:
record.name = "%s (%s)" % (record.partner_id.name, record.amount)

View File

@@ -0,0 +1,13 @@
from odoo import _, api, fields, models
class ProjectTask(models.Model):
_inherit = "project.task"
funder_ids = fields.One2many("project.funder", "task_id", string="funder")
amount_total = fields.Float(compute="_compute_amount_total", string="Amount Total")
@api.depends("funder_ids")
def _compute_amount_total(self):
for record in self:
record.amount_total = sum(record.funder_ids.mapped("amount"))

View File

@@ -0,0 +1,2 @@
id,name,model_id/id,group_id/id,perm_read,perm_write,perm_create,perm_unlink
access_project_funder_all,project.funder.all,model_project_funder,,1,1,1,1
1 id name model_id/id group_id/id perm_read perm_write perm_create perm_unlink
2 access_project_funder_all project.funder.all model_project_funder 1 1 1 1

View File

@@ -0,0 +1,47 @@
<?xml version="1.0" encoding="UTF-8"?>
<odoo>
<record id="view_task_form2_inherit_project" model="ir.ui.view">
<field name="name">project.task.view.form.inherit</field>
<field name="model">project.task</field>
<field name="inherit_id" ref="project.view_task_form2"/>
<field name="arch" type="xml">
<xpath expr="//notebook" position="inside">
<page name="funders" string="Funder(s)">
<field name="funder_ids" context="{'default_task_id': active_id}">
<tree editable="bottom">
<field name="task_id" invisible="1"/>
<field name="partner_id"/>
<field name="amount"/>
</tree>
</field>
<group>
<group class="oe_subtotal_footer oe_right" name="project_hours">
<span>
<label class="font-weight-bold" for="amount_total" string="Total"/>
</span>
<field name="amount_total" nolabel="1"/>
</group>
</group>
</page>
</xpath>
</field>
</record>
<record id="view_task_kanban_inherit_project_funders" model="ir.ui.view">
<field name="name">project.task.view.kanban.inherit</field>
<field name="model">project.task</field>
<field name="inherit_id" ref="project.view_task_kanban"/>
<field name="arch" type="xml">
<field name="active" position="after">
<field name="amount_total"/>
</field>
<field name="tag_ids" position="after">
<span attrs="{'invisible': [('amount_total', '=', 0)]}">
Funds: <field name="amount_total"/>
</span>
</field>
</field>
</record>
</odoo>

View File

@@ -1,60 +0,0 @@
# Project Link From Invoice
![License: AGPL-3](https://img.shields.io/badge/licence-AGPL--3-blue.svg)
Easily access your projects directly from invoices:
- Displays project name(s) in the invoice list view.
- Adds a smart button on invoice form view to open the related project(s).
## Installation
This module depends on:
- `project` (Odoo core)
- `account` (Odoo core)
- `sale_project` (Odoo core)
Use Odoo's normal procedure to install add-ons.
## Configuration
No specific configuration is required.
## Usage
Open an invoice that is linked to a sale order. If the sale order has tasks
in a project, a "Project(s)" button appears in the header of the invoice form.
Clicking it opens the related project(s) in form or list view.
The invoice list view also displays the project name(s) in a dedicated column.
## Known issues / Roadmap
None yet.
## Bug Tracker
Bugs are tracked on [our issues website](https://github.com/elabore-coop/project-tools/issues).
In case of trouble, please check there if your issue has already been reported.
If you spotted it first, help us smash it by providing detailed and welcomed feedback.
## Credits
### Images
- Elabore: [Icon](https://elabore.coop/web/image/res.company/1/logo)
### Contributors
- Clément Thomas
### Funders
The development of this module has been financially supported by:
- Elabore (<https://elabore.coop>)
### Maintainer
This module is maintained by Elabore.

View File

@@ -0,0 +1,47 @@
=================
project_link_from_invoice
=================
Easily access to your projects from invoices
View project(s) in invoices tree view
Access to related projects of an invoice from a button
Installation
============
Use Odoo normal procedure to install add-ons to install
``project_link_from_invoice``.
Known issues / Roadmap
======================
Bug Tracker
===========
Bugs are tracked on `our issues website
<https://github.com/elabore-coop/project-tools/issues>`_. In case of
trouble, please check there if your issue has already been
reported. If you spotted it first, help us smashing it by providing a
detailed and welcomed feedback.
Credits
=======
Images
------
* Elabore: `Icon <https://elabore.coop/web/image/res.company/1/logo?unique=f3db262>`_.
Contributors
------------
* Clément Thomas
Funders
-------
The development of this module has been financially supported by:
* Elabore (https://elabore.coop)
Maintainer
----------
This module is maintained by Elabore.

View File

@@ -3,19 +3,76 @@
{
"name": "project_link_from_invoice",
"version": "18.0.1.0.0",
"version": "16.0.1.0.0",
"author": "Elabore",
"website": "https://git.elabore.coop/elabore/project-tools",
"website": "https://github.com/elabore-coop/project-tools",
"maintainer": "Clément Thomas",
"license": "AGPL-3",
"category": "Tools",
"summary": "Add link from invoice to project",
"description": """
:image: https://img.shields.io/badge/licence-AGPL--3-blue.svg
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
:alt: License: AGPL-3
=================
project_link_from_invoice
=================
Users can access to related project from invoices
Installation
============
Install ``project_link_from_invoice``, all dependencies will be installed by default.
Known issues / Roadmap
======================
None yet.
Bug Tracker
===========
Bugs are tracked on `our issues website
<https://github.com/elabore-coop/project-tools/issues>`_. In case of
trouble, please check there if your issue has already been
reported. If you spotted it first, help us smashing it by providing a
detailed and welcomed feedback.
Credits
=======
Images
------
* Elabore: `Icon <https://elabore.coop/web/image/res.company/1/logo?unique=f3db262>`_.
Contributors
------------
* Clément Thomas <https://github.com/stephansainleger>
Funders
-------
The development of this module has been financially supported by:
* Elabore (https://elabore.coop)
Maintainer
----------
This module is maintained by Elabore.
""",
# any module necessary for this one to work correctly
"depends": [
"base",
"project",
"account",
"sale_project",
"sale"
],
"qweb": [
# "static/src/xml/*.xml",
],
"external_dependencies": {
"python": [],
@@ -26,6 +83,8 @@
],
# only loaded in demonstration mode
"demo": [],
"js": [],
"css": [],
"installable": True,
# Install this module automatically if all dependency have been previously
# and independently installed. Used for synergetic or glue modules.

View File

@@ -1,49 +1,38 @@
# Copyright 2022 Stéphan Sainléger (Elabore)
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from odoo import api, fields, models
from odoo import models, fields, api
class AccountMove(models.Model):
_inherit = "account.move"
project_ids = fields.Many2many(
"project.project", string="Projects", compute="_get_related_project_ids"
)
project_count = fields.Integer("Project Count", compute="_get_related_project_ids")
projects_name = fields.Char("Project(s)", compute="_get_related_project_ids")
project_ids = fields.Many2many('project.project', name="Projects", compute='get_related_project_ids')
project_count = fields.Integer("Project Count", compute='get_related_project_ids')
projects_name = fields.Char('Project(s)', compute='get_related_project_ids')
def action_open_projects(self):
"""Open related projects, in form or list view depending on project numbers."""
'''
Open related projects, in form or tree view depending on project numbers
'''
project_ids = self.project_ids.ids
action = self.env["ir.actions.actions"]._for_xml_id(
"project.open_view_project_all"
)
action = self.env["ir.actions.actions"]._for_xml_id("project.open_view_project_all")
if self.project_count == 1:
action["res_id"] = project_ids[0]
action["views"] = [[False, "form"]]
action['res_id'] = project_ids[0]
action['views'] = [[False, "form"]]
else:
action["views"] = [[False, "list"], [False, "form"]]
action["domain"] = [("id", "in", project_ids)]
del action["target"] # to display breadcrumbs
action['views'] = [[False, "tree"], [False, "form"]]
action['domain'] = [('id', 'in', project_ids)]
del action['target'] #to display breadcrumbs
return action
@api.depends("line_ids.sale_line_ids")
def _get_related_project_ids(self):
@api.depends('line_ids.sale_line_ids')
def get_related_project_ids(self):
for move in self:
projects = self.env["project.task"].search(
[
(
"sale_order_id",
"in",
move.line_ids.sale_line_ids.order_id.ids,
)
]
).project_id
projects = self.env['project.task'].search([('sale_order_id','in',move.line_ids.sale_line_ids.order_id.ids)]).project_id
move.project_ids = projects.ids
move.projects_name = " ; ".join([p.name for p in projects])
move.project_count = len(projects)
move.projects_name = ' ; '.join([p.name for p in projects])
move.project_count = len(projects)

View File

@@ -1,4 +0,0 @@
# Copyright 2022 Stéphan Sainléger (Elabore)
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from . import test_account_move

View File

@@ -1,193 +0,0 @@
# Copyright 2022 Stéphan Sainléger (Elabore)
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from odoo import Command
from odoo.tests import tagged
from odoo.addons.sale_project.tests.common import TestSaleProjectCommon
@tagged("post_install", "-at_install")
class TestProjectLinkFromInvoice(TestSaleProjectCommon):
"""Tests for the project_link_from_invoice module.
Verifies that invoices correctly expose related project(s) via
the computed fields project_ids, project_count and projects_name.
"""
@classmethod
def setUpClass(cls):
super().setUpClass()
cls.env = cls.env(context=dict(cls.env.context, tracking_disable=True))
# Service product that creates a task in a new project on SO confirmation
cls.service_product = cls.env["product.product"].create(
{
"name": "Service with Project",
"type": "service",
"invoice_policy": "order",
"service_tracking": "task_in_project",
}
)
# Partner used for orders and invoices
cls.partner = cls.env["res.partner"].create({"name": "Test Partner"})
def _confirm_sale_order(self, product, qty=1):
"""Create and confirm a sale order with one service line."""
so = self.env["sale.order"].create(
{
"partner_id": self.partner.id,
"order_line": [
Command.create(
{
"product_id": product.id,
"product_uom_qty": qty,
}
)
],
}
)
so.action_confirm()
return so
def _invoice_sale_order(self, sale_order):
"""Create and post an invoice from a confirmed sale order."""
invoice_wizard = (
self.env["sale.advance.payment.inv"]
.with_context(
active_model="sale.order",
active_ids=sale_order.ids,
active_id=sale_order.id,
)
.create({"advance_payment_method": "delivered"})
)
invoices = invoice_wizard._create_invoices(sale_order)
invoices.action_post()
return invoices
# ------------------------------------------------------------------
# Tests
# ------------------------------------------------------------------
def test_no_project_on_unrelated_invoice(self):
"""An invoice not linked to any sale order has no related project."""
invoice = self.env["account.move"].create(
{
"move_type": "out_invoice",
"partner_id": self.partner.id,
"invoice_line_ids": [
Command.create(
{
"name": "Manual line",
"quantity": 1,
"price_unit": 100.0,
}
)
],
}
)
self.assertEqual(invoice.project_count, 0)
self.assertFalse(invoice.project_ids)
self.assertEqual(invoice.projects_name, "")
def test_single_project_linked_to_invoice(self):
"""An invoice from a SO that created a task has one related project."""
so = self._confirm_sale_order(self.service_product)
# The SO must have generated a task in a project
task = so.tasks_ids[:1]
self.assertTrue(task, "Sale order confirmation should have created a task")
project = task.project_id
self.assertTrue(project, "Task should belong to a project")
invoice = self._invoice_sale_order(so)
self.assertEqual(len(invoice), 1)
self.assertEqual(invoice.project_count, 1)
self.assertIn(project, invoice.project_ids)
self.assertIn(project.name, invoice.projects_name)
def test_multiple_projects_linked_to_invoice(self):
"""An invoice from two SOs with different projects shows both projects."""
so1 = self._confirm_sale_order(self.service_product)
so2 = self._confirm_sale_order(self.service_product)
project1 = so1.tasks_ids[:1].project_id
project2 = so2.tasks_ids[:1].project_id
self.assertTrue(project1 and project2)
# Ensure the two SOs created different projects
self.assertNotEqual(project1, project2)
# Create a single invoice covering both sale orders
invoice_wizard = (
self.env["sale.advance.payment.inv"]
.with_context(
active_model="sale.order",
active_ids=(so1 + so2).ids,
)
.create({"advance_payment_method": "delivered"})
)
invoices = invoice_wizard._create_invoices(so1 + so2)
invoices.action_post()
# Find the invoice that covers both SO lines
combined_invoice = invoices[:1]
self.assertGreaterEqual(combined_invoice.project_count, 2)
self.assertIn(project1, combined_invoice.project_ids)
self.assertIn(project2, combined_invoice.project_ids)
def test_action_open_projects_single(self):
"""action_open_projects returns a form view when there is one project."""
so = self._confirm_sale_order(self.service_product)
invoice = self._invoice_sale_order(so)
self.assertEqual(invoice.project_count, 1)
action = invoice.action_open_projects()
self.assertEqual(action["res_id"], invoice.project_ids.id)
view_types = [v[1] for v in action["views"]]
self.assertIn("form", view_types)
self.assertNotIn("list", view_types)
def test_action_open_projects_multiple(self):
"""action_open_projects returns a list view when there are multiple projects."""
so1 = self._confirm_sale_order(self.service_product)
so2 = self._confirm_sale_order(self.service_product)
invoice_wizard = (
self.env["sale.advance.payment.inv"]
.with_context(
active_model="sale.order",
active_ids=(so1 + so2).ids,
)
.create({"advance_payment_method": "delivered"})
)
invoices = invoice_wizard._create_invoices(so1 + so2)
invoices.action_post()
combined_invoice = invoices[:1]
if combined_invoice.project_count < 2:
self.skipTest("Combined invoice does not span multiple projects")
action = combined_invoice.action_open_projects()
view_types = [v[1] for v in action["views"]]
self.assertIn("list", view_types)
self.assertIn("form", view_types)
self.assertNotIn("target", action)
def test_projects_name_separator(self):
"""projects_name joins multiple project names with ' ; '."""
so1 = self._confirm_sale_order(self.service_product)
so2 = self._confirm_sale_order(self.service_product)
invoice_wizard = (
self.env["sale.advance.payment.inv"]
.with_context(
active_model="sale.order",
active_ids=(so1 + so2).ids,
)
.create({"advance_payment_method": "delivered"})
)
invoices = invoice_wizard._create_invoices(so1 + so2)
invoices.action_post()
combined_invoice = invoices[:1]
if combined_invoice.project_count >= 2:
self.assertIn(" ; ", combined_invoice.projects_name)

View File

@@ -6,26 +6,23 @@
<field name="inherit_id" ref="account.view_move_form" />
<field name="priority" eval="99" />
<field name="arch" type="xml">
<xpath expr="//div[@name='button_box']" position="inside">
<button
name="action_open_projects"
class="oe_stat_button"
icon="fa-puzzle-piece"
type="object"
invisible="project_count == 0"
>
<field name="project_count" widget="statinfo" string="Project(s)" />
</button>
<xpath expr="//header" position="inside">
<button name="action_open_projects"
string="Project(s)" class="oe_highlight" type="object" icon="fa-puzzle-piece"
invisible="[]" attrs="{'invisible': [('project_count', '=', 0)]}" />
</xpath>
<xpath expr="//header" position="after">
<field name="project_count" invisible="1" />
</xpath>
</field>
</record>
<record id="view_invoice_tree_project_link_from_invoice_inherit" model="ir.ui.view">
<field name="name">account.invoice.list.project.link</field>
<field name="name">account.invoice.tree.project.link</field>
<field name="model">account.move</field>
<field name="inherit_id" ref="account.view_invoice_tree"/>
<field name="arch" type="xml">
<xpath expr="//list/field[@name='invoice_date']" position="before">
<xpath expr="//tree/field[@name='invoice_date']" position="before">
<field name="projects_name" />
</xpath>
</field>

2
project_name_from_lead/.gitignore vendored Normal file
View File

@@ -0,0 +1,2 @@
*.*~
*pyc

View File

@@ -0,0 +1,57 @@
=======================
project_name_from_quote
=======================
.. |badge1| image:: https://img.shields.io/badge/maturity-Beta-yellow.png
:target: https://odoo-community.org/page/development-status
:alt: Beta
.. |badge2| image:: https://img.shields.io/badge/licence-AGPL--3-blue.png
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
:alt: License: AGPL-3
.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Faccount--analytic-lightgray.png?logo=github
:target: https://github.com/elabore-coop/project-tools
:alt: elabore/hr-tools
|badge1| |badge2| |badge3|
This module tranfers the Quote / Sale order title to the project name.
Installation
============
Use Odoo normal module installation procedure to install ``project_name_from_quote``.
Known issues / Roadmap
======================
None yet.
Bug Tracker
===========
Bugs are tracked on `our issues website <https://github.com/elabore-coop/project-tools/issues>`_. In case of
trouble, please check there if your issue has already been
reported. If you spotted it first, help us smashing it by providing a
detailed and welcomed feedback.
Credits
=======
Contributors
------------
* Stéphan Sainléger
Funders
-------
The development of this module has been financially supported by:
* Elabore (https://elabore.coop)
* Datactivist (https://datactivist.coop)
Maintainer
----------
This module is maintained by Elabore.

View File

@@ -0,0 +1,2 @@
from . import models

View File

@@ -1,19 +1,20 @@
# Copyright 2022 Laetitia Da Costa (Elabore)
# Copyright 2022 Stéphan Sainléger (Elabore)
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
{
"name": "project_only_current_project_tasks_as_parent",
"version": "18.0.1.0.0",
"name": "project_name_from_lead",
"version": "16.0.1.0.0",
"author": "Elabore",
"website": "https://git.elabore.coop/elabore/project-tools",
"maintainer": "Laetitia Da Costa",
"website": "https://elabore.coop",
"maintainer": "Clément Thomas",
"license": "AGPL-3",
"category": "Project",
"summary": "In parent's tasks dropdown field, show only tasks from the current projet",
"category": "Tools",
"summary": "Use lead name as project name",
# any module necessary for this one to work correctly
"depends": [
"base",
"project",
"crm",
"sale_project",
],
"qweb": [
# "static/src/xml/*.xml",
@@ -22,9 +23,7 @@
"python": [],
},
# always loaded
"data": [
"views/project_task.xml",
],
"data": [],
# only loaded in demonstration mode
"demo": [],
"js": [],
@@ -34,4 +33,4 @@
# and independently installed. Used for synergetic or glue modules.
"auto_install": False,
"application": False,
}
}

View File

@@ -0,0 +1 @@
from . import sale_order, crm_lead

View File

@@ -0,0 +1,17 @@
from odoo import models
class Lead(models.Model):
_inherit = 'crm.lead'
def write(self, vals):
"""update project name if project created from lead
"""
for lead in self:
if 'name' in vals:
sale = self.env['sale.order'].search([('opportunity_id','=',lead.id)])
if sale:
project = self.env['project.project'].search([('sale_order_id','=',sale.id)])
if project:
project.name = vals['name']
return super(Lead, self).write(vals)

View File

@@ -0,0 +1,12 @@
from typing import ValuesView
from odoo import models
class SaleOrderLine(models.Model):
_inherit = "sale.order.line"
def _timesheet_create_project_prepare_values(self):
values = super(SaleOrderLine, self)._timesheet_create_project_prepare_values()
if self.order_id and self.order_id.opportunity_id:
values['name'] = self.order_id.opportunity_id.name
return values

2
project_name_from_quote/.gitignore vendored Normal file
View File

@@ -0,0 +1,2 @@
*.*~
*pyc

View File

@@ -0,0 +1,57 @@
=======================
project_name_from_quote
=======================
.. |badge1| image:: https://img.shields.io/badge/maturity-Beta-yellow.png
:target: https://odoo-community.org/page/development-status
:alt: Beta
.. |badge2| image:: https://img.shields.io/badge/licence-AGPL--3-blue.png
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
:alt: License: AGPL-3
.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Faccount--analytic-lightgray.png?logo=github
:target: https://github.com/elabore-coop/project-tools
:alt: elabore/hr-tools
|badge1| |badge2| |badge3|
This module tranfers the Quote / Sale order title to the project name.
Installation
============
Use Odoo normal module installation procedure to install ``project_name_from_quote``.
Known issues / Roadmap
======================
None yet.
Bug Tracker
===========
Bugs are tracked on `our issues website <https://github.com/elabore-coop/project-tools/issues>`_. In case of
trouble, please check there if your issue has already been
reported. If you spotted it first, help us smashing it by providing a
detailed and welcomed feedback.
Credits
=======
Contributors
------------
* Stéphan Sainléger
Funders
-------
The development of this module has been financially supported by:
* Elabore (https://elabore.coop)
* Datactivist (https://datactivist.coop)
Maintainer
----------
This module is maintained by Elabore.

View File

@@ -0,0 +1,2 @@
from . import models

View File

@@ -0,0 +1,36 @@
# Copyright 2022 Stéphan Sainléger (Elabore)
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
{
"name": "project_name_from_quote",
"version": "16.0.1.0.0",
"author": "Elabore",
"website": "https://elabore.coop",
"maintainer": "Stéphan Sainléger",
"license": "AGPL-3",
"category": "Tools",
"summary": "Use quote name as project name",
# any module necessary for this one to work correctly
"depends": [
"base",
"sale_project",
"account_quotation_sale_order_invoice_title",
],
"qweb": [
# "static/src/xml/*.xml",
],
"external_dependencies": {
"python": [],
},
# always loaded
"data": [],
# only loaded in demonstration mode
"demo": [],
"js": [],
"css": [],
"installable": True,
# Install this module automatically if all dependency have been previously
# and independently installed. Used for synergetic or glue modules.
"auto_install": False,
"application": False,
}

View File

@@ -0,0 +1 @@
from . import sale_order

View File

@@ -0,0 +1,11 @@
from typing import ValuesView
from odoo import models
class SaleOrderLine(models.Model):
_inherit = "sale.order.line"
def _timesheet_create_project_prepare_values(self):
values = super(SaleOrderLine, self)._timesheet_create_project_prepare_values()
values["name"] = self.order_id.so_title
return values

View File

@@ -0,0 +1,86 @@
# Copyright 2022 Laetitia Da Costa (Elabore)
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
{
"name": "project_parent_tasks",
"version": "14.0.1.0.0",
"author": "Elabore",
"website": "https://elabore.coop",
"maintainer": "Laetitia Da Costa",
"license": "AGPL-3",
"category": "Project",
"summary": "in parent's tasks dropdown field, show only tasks from the current projet",
"description": """
:image: https://img.shields.io/badge/licence-AGPL--3-blue.svg
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
:alt: License: AGPL-3
====================
project_parent_tasks
====================
when selecting a parent task in a task, display only tasks from the current project
Installation
============
Use Odoo normal module installation procedure to install ``project_parent_tasks``.
Known issues / Roadmap
======================
None yet.
Bug Tracker
===========
Bugs are tracked on `our issues website <https://github.com/elabore-coop/project-tools/issues>`_. In case of
trouble, please check there if your issue has already been
reported. If you spotted it first, help us smashing it by providing a
detailed and welcomed feedback.
Credits
=======
Contributors
------------
* Laetitia Da Costa (https://github.com/LaetitiaElabore)
Funders
-------
The development of this module has been financially supported by:
* Elabore (https://elabore.coop)
Maintainer
----------
This module is maintained by Elabore.
""",
# any module necessary for this one to work correctly
"depends": [
"base",
"project",
],
"qweb": [
# "static/src/xml/*.xml",
],
"external_dependencies": {
"python": [],
},
# always loaded
"data": [
"views/project_task.xml",
],
# only loaded in demonstration mode
"demo": [],
"js": [],
"css": [],
"installable": True,
# Install this module automatically if all dependency have been previously
# and independently installed. Used for synergetic or glue modules.
"auto_install": False,
"application": False,
}

View File

@@ -1,18 +1,13 @@
<?xml version="1.0" encoding="UTF-8" ?>
<?xml version="1.0" encoding="UTF-8"?>
<odoo>
<record model="ir.ui.view" id="view_tasks_form2_parent_project_inherited">
<field name="name">view.tasks.form2.parent.project.inherited</field>
<field name="model">project.task</field>
<field name="inherit_id" ref="project.view_task_form2" />
<field name="arch" type="xml">
<xpath
expr="//page[@name='extra_info']//field[@name='parent_id']"
position="attributes"
>
<attribute
name="domain"
>[('project_id','=', project_id), ('stage_id.fold', '=', False)]</attribute>
<xpath expr="//page[@name='extra_info']//field[@name='parent_id']" position="attributes">
<attribute name="domain">[('project_id','=', project_id)]</attribute>
</xpath>
</field>
</record>
</odoo>
</odoo>

View File

@@ -3,9 +3,9 @@
{
"name": "project_request_data",
"version": "18.0.1.0.0",
"version": "16.0.1.2.0",
"author": "Elabore",
"website": "https://git.elabore.coop/elabore/project-tools",
"website": "https://elabore.coop",
"maintainer": "Stéphan Sainléger",
"license": "AGPL-3",
"category": "Project",
@@ -89,4 +89,4 @@ This module is maintained by Elabore.
# and independently installed. Used for synergetic or glue modules.
"auto_install": False,
"application": False,
}
}

View File

@@ -1,3 +1,3 @@
from . import task_service
from . import request_type
from . import project_task
from . import project_task

View File

@@ -1,8 +1,9 @@
from odoo import fields, models
from odoo import models, fields
class Task(models.Model):
_inherit = "project.task"
service_id = fields.Many2one("task.service", string="Service")
request_type_id = fields.Many2one("request.type", string="Request Type")
service_id = fields.Many2one('task.service', string='Service')
request_type_id = fields.Many2one('request.type', string='Request Type')

View File

@@ -1,9 +1,9 @@
from odoo import fields, models
from odoo import models, fields
class RequestType(models.Model):
_name = "request.type"
_description = "Request Type"
name = fields.Char("name", required=True)
sequence = fields.Integer()
name = fields.Char('name', required=True)
sequence = fields.Integer()

View File

@@ -1,9 +1,9 @@
from odoo import fields, models
from odoo import models, fields
class TaskService(models.Model):
_name = "task.service"
_description = "Task service"
name = fields.Char("name", required=True)
sequence = fields.Integer()
name = fields.Char('name', required=True)
sequence = fields.Integer()

View File

@@ -2,4 +2,4 @@ id,name,model_id/id,group_id/id,perm_read,perm_write,perm_create,perm_unlink
access_task_service_user,project.task_service.user,model_task_service,,1,0,0,0
access_task_service_manager,project.task_service.manager,model_task_service,project.group_project_manager,1,1,1,1
access_request_type_user,project.request_type.user,model_request_type,,1,0,0,0
access_request_type_manager,project.request_type.manager,model_request_type,project.group_project_manager,1,1,1,1
access_request_type_manager,project.request_type.manager,model_request_type,project.group_project_manager,1,1,1,1
1 id name model_id/id group_id/id perm_read perm_write perm_create perm_unlink
2 access_task_service_user project.task_service.user model_task_service 1 0 0 0
3 access_task_service_manager project.task_service.manager model_task_service project.group_project_manager 1 1 1 1
4 access_request_type_user project.request_type.user model_request_type 1 0 0 0
5 access_request_type_manager project.request_type.manager model_request_type project.group_project_manager 1 1 1 1

View File

@@ -1,15 +1,7 @@
<?xml version="1.0" encoding="utf-8" ?>
<?xml version="1.0" encoding="utf-8"?>
<odoo>
<template
id="portal_my_task_request_data"
name="My Task: Request Data"
inherit_id="project.portal_my_task"
priority="40"
>
<xpath
expr="//div[@id='card_body']/div[hasclass('row','mb-4','container')]"
position="after"
>
<template id="portal_my_task_request_data" name="My Task: Request Data" inherit_id="project.portal_my_task" priority="40">
<xpath expr="//div[@id='card_body']/div[hasclass('row','mb-4','container')]" position="after">
<div id="request_data" class="row mb-4">
<div class="col-12 col-md-6" t-if="task.service_id">
<strong>Service:</strong>
@@ -22,4 +14,4 @@
</div>
</xpath>
</template>
</odoo>
</odoo>

View File

@@ -1,4 +1,4 @@
<?xml version="1.0" encoding="UTF-8" ?>
<?xml version="1.0" encoding="UTF-8"?>
<odoo>
<record id="view_task_form2_request_data" model="ir.ui.view">
<field name="name">project.task.form.request.data</field>
@@ -18,19 +18,12 @@
<field name="model">project.task</field>
<field name="arch" type="xml">
<xpath expr="//field[@name='name']" position="before">
<div style="font-size:11px; margin-bottom: 8px;"><field
name="create_date"
widget="date"
/></div>
<div style="font-size:11px; margin-bottom: 8px;"><field name="create_date" widget="date"/></div>
</xpath>
<xpath expr="//field[@name='tag_ids']" position="before">
<span style="background:lightblue; font-size:11px"><field
name="service_id"
/></span>
<span style="background:lightsteelblue; font-size:11px"><field
name="request_type_id"
/></span>
<span style="background:lightblue; font-size:11px"><field name="service_id"/></span>
<span style="background:lightsteelblue; font-size:11px"><field name="request_type_id"/></span>
</xpath>
</field>
</record>
</odoo>
</odoo>

View File

@@ -1,29 +1,23 @@
<?xml version="1.0" encoding="utf-8" ?>
<?xml version="1.0" encoding="utf-8"?>
<odoo>
<record id="request_type_view_tree" model="ir.ui.view">
<field name="name">request.type.view.list</field>
<field name="name">request.type.view.tree</field>
<field name="model">request.type</field>
<field name="arch" type="xml">
<list editable="top" default_order="sequence">
<tree string="Request Types" editable="top" default_order="sequence">
<field name="sequence" widget="handle" />
<field name="name" />
</list>
</tree>
</field>
</record>
<record id="act_request_types_list" model="ir.actions.act_window">
<field name="name">Request Types</field>
<field name="res_model">request.type</field>
<field name="view_mode">list</field>
<field name="view_mode">tree</field>
</record>
<menuitem
id="menu_request_types"
action="act_request_types_list"
parent="project.menu_project_config"
sequence="99"
name="Request Types"
/>
<menuitem id="menu_request_types" action="act_request_types_list" parent="project.menu_project_config" sequence="99" name="Request Types" />
</odoo>
</odoo>

View File

@@ -1,29 +1,23 @@
<?xml version="1.0" encoding="utf-8" ?>
<?xml version="1.0" encoding="utf-8"?>
<odoo>
<record id="task_service_view_tree" model="ir.ui.view">
<field name="name">task.service.view.list</field>
<field name="name">task.service.view.tree</field>
<field name="model">task.service</field>
<field name="arch" type="xml">
<list editable="top" default_order="sequence">
<tree string="Task Services" editable="top" default_order="sequence">
<field name="sequence" widget="handle" />
<field name="name" />
</list>
</tree>
</field>
</record>
<record id="act_task_services_list" model="ir.actions.act_window">
<field name="name">Task Services</field>
<field name="res_model">task.service</field>
<field name="view_mode">list</field>
<field name="view_mode">tree</field>
</record>
<menuitem
id="menu_task_services"
action="act_task_services_list"
parent="project.menu_project_config"
sequence="99"
name="Task Services"
/>
<menuitem id="menu_task_services" action="act_task_services_list" parent="project.menu_project_config" sequence="99" name="Task Services" />
</odoo>
</odoo>

View File

@@ -3,17 +3,22 @@
{
"name": "Project task billable hours",
"version": "18.0.1.0.0",
"version": "16.0.1.0.0",
"author": "Elabore",
"website": "https://git.elabore.coop/elabore/project-tools",
"website": "https://github.com/elabore-coop/project-tools",
"maintainer": "Laetitia Da Costa",
"license": "AGPL-3",
"category": "Tools",
"summary": "In task kanban view, display remaining billable hours instead of remaining hours",
"description": "",
"depends": ["project", "hr_timesheet", "project_working_time_task_portal"],
"depends": [
"project",
"hr_timesheet",
"project_working_time_task_portal"
],
"data": [
"views/project_task_views.xml",
"views/project_views.xml"
],
"installable": True,
"application": False,

View File

@@ -0,0 +1 @@
from . import billable_time

View File

@@ -0,0 +1,16 @@
from odoo import models, fields, api
class Project(models.Model):
_inherit = "project.project"
billable_remaining_hours = fields.Float(
compute="_compute_project_billable_remaining_hours",
string="Billable Remaining Hours",
store=True,
help="Total Billable remaining time (without exclude_from_sale_order timesheet lines)."
)
@api.depends("task_ids.billable_remaining_hours")
def _compute_project_billable_remaining_hours(self):
for project in self:
project.billable_remaining_hours = sum(task.billable_remaining_hours for task in project.task_ids)

View File

@@ -1,39 +1,21 @@
<?xml version="1.0" encoding="UTF-8" ?>
<?xml version="1.0" encoding="UTF-8"?>
<odoo>
<!-- Replace remaining hours by billable remaining hours in kanban view -->
<record id="view_task_kanban_billable_remaining_hours_inherit" model="ir.ui.view">
<field name="name">project.task.kanban.billable.remaining.hours.inherit</field>
<field name="model">project.task</field>
<field
name="inherit_id"
ref="hr_timesheet.view_task_kanban_inherited_progress"
/>
<field name="inherit_id" ref="hr_timesheet.view_task_kanban_inherited_progress"/>
<field name="arch" type="xml">
<xpath expr="//t[@name='allocated_hours']" position="replace">
<t t-set="badge" t-value="" />
<t
t-set="badge"
t-value="'badge-warning'"
t-if="record.billable_progress &gt;= 80 and record.billable_progress &lt;= 100"
/>
<t
t-set="badge"
t-value="'badge-danger'"
t-if="record.billable_remaining_hours.raw_value &lt; 0"
/>
<t
t-set="title"
t-value="'Remaining days'"
t-if="record.encode_uom_in_days.raw_value"
/>
<t t-set="title" t-value="'Billable remaining hours'" t-else="" />
<div
t-attf-class="oe_kanban_align badge {{ badge }}"
t-att-title="title"
>
<xpath expr="//t[@name='planned_hours']" position="replace">
<t t-set="badge" t-value=""/>
<t t-set="badge" t-value="'badge-warning'" t-if="record.billable_progress &gt;= 80 and record.billable_progress &lt;= 100"/>
<t t-set="badge" t-value="'badge-danger'" t-if="record.billable_remaining_hours.raw_value &lt; 0"/>
<t t-set="title" t-value="'Remaining days'" t-if="record.encode_uom_in_days.raw_value"/>
<t t-set="title" t-value="'Billable remaining hours'" t-else=""/>
<div t-attf-class="oe_kanban_align badge {{ badge }}" t-att-title="title">
<field name="billable_remaining_hours" widget="timesheet_uom" />
</div>
</xpath>
</field>
</record>
</odoo>
</odoo>

View File

@@ -0,0 +1,19 @@
<?xml version="1.0" encoding="UTF-8"?>
<odoo>
<record id="project_working_time_view_project_kanban" model="ir.ui.view">
<field name="name">project.working.time.view.project.kanban</field>
<field name="model">project.project</field>
<field name="inherit_id" ref="hr_timesheet.view_project_kanban_inherited" />
<field name="priority" eval="999" />
<field name="arch" type="xml">
<xpath expr="//div[hasclass('o_project_kanban_boxes')]/following-sibling::div[1]" position="replace">
<t t-set="badgeColor" t-value="'border-success'"/>
<t t-set="badgeColor" t-value="'border-danger'" t-if="record.billable_remaining_hours.raw_value &lt; 0"/>
<div t-if="record.allow_timesheets.raw_value and record.allocated_hours.raw_value &gt; 0"
t-attf-class="oe_kanban_align badge border {{ badgeColor }}" title="Billable remaining hours" groups="hr_timesheet.group_hr_timesheet_user">
<field name="billable_remaining_hours" widget="timesheet_uom"/>
</div>
</xpath>
</field>
</record>
</odoo>

2
project_task_portal_form/.gitignore vendored Normal file
View File

@@ -0,0 +1,2 @@
*.*~
*pyc

View File

@@ -0,0 +1,7 @@
=====================
project_task_portal_form
=====================
Add a portal form to create project tasks
This is an Odoo addon.

View File

@@ -0,0 +1,3 @@
# -*- coding: utf-8 -*-
from . import controllers

View File

@@ -0,0 +1,86 @@
# Copyright 2022 Stéphan Sainléger (Elabore)
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
{
"name": "project_task_portal_form",
"version": "16.0.1.1.0",
"author": "Elabore",
"website": "https://elabore.coop",
"maintainer": "Stéphan Sainléger",
"license": "AGPL-3",
"category": "Tools",
"summary": "Add a portal form to create project tasks",
"description": """
:image: https://img.shields.io/badge/licence-AGPL--3-blue.svg
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
:alt: License: AGPL-3
========================
project_task_portal_form
========================
Add a portal form to create project tasks
Installation
============
Use Odoo normal module installation procedure to install
``project_task_portal_form``.
Known issues / Roadmap
======================
None yet.
Bug Tracker
===========
Bugs are tracked on `our issues website <https://github.com/elabore-coop/project-tools/issues>`_. In case of
trouble, please check there if your issue has already been
reported. If you spotted it first, help us smashing it by providing a
detailed and welcomed feedback.
Credits
=======
Contributors
------------
* Stéphan Sainléger
Funders
-------
The development of this module has been financially supported by:
* Elabore (https://elabore.coop)
Maintainer
----------
This module is maintained by Elabore.
""",
# any module necessary for this one to work correctly
"depends": [
"base",
"project",
"project_request_data",
"project_user_default_project",
],
"qweb": [],
"external_dependencies": {
"python": [],
},
# always loaded
"data": [
"views/portal_task_creation_form.xml",
],
# only loaded in demonstration mode
"demo": [],
"js": [],
"css": [],
"installable": True,
# Install this module automatically if all dependency have been previously
# and independently installed. Used for synergetic or glue modules.
"auto_install": False,
"application": False,
}

View File

@@ -0,0 +1,3 @@
# -*- coding: utf-8 -*-
from . import portal_task_creation

View File

@@ -0,0 +1,127 @@
# Copyright 2020 Lokavaluto ()
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
import base64
from odoo import http
from odoo.http import request
from odoo.addons.portal.controllers.portal import CustomerPortal
class PortalTaskCreation(CustomerPortal):
_TASK_CREATION_FIELDS = [
"name",
"service_id",
"request_type_id",
"small_description", #not in v14
"access", #not in v14
"bug_report", #not in v14
"priority",
]
# Variable to update to add other fields in child classes
_EXTRA_FIELDS = []
def _taskform_get_page_view_values(self, partner, access_token, **kwargs):
values = {
"page_name": "portal_task_form",
"partner": partner,
}
return self._get_page_view_values(
partner,
access_token,
values,
"my_task_creation_history",
False,
**kwargs
)
def _get_task_priorities(self):
priorities = []
for id, name in request.env['project.task']._fields['priority'].selection:
value = {
"id": id,
"name": name
}
priorities.append(value)
return priorities
@http.route(
["/task/form"],
type="http",
auth="user",
website=True,
)
def portal_task_creation(self, access_token=None, redirect=None, **kw):
values = self._taskform_get_page_view_values(request.env.user.partner_id, access_token, **kw)
request_types = request.env["request.type"].sudo().search([])
task_services = request.env["task.service"].sudo().search([])
priorities = self._get_task_priorities()
error = dict()
error_message = []
values.update(
{
"request_types": request_types,
"task_services": task_services,
"priorities": priorities,
"error": error,
"error_message": error_message,
}
)
return request.render("project_task_portal_form.portal_task_creation_form", values)
def _compute_form_data(self, data):
values = {}
for field in self._TASK_CREATION_FIELDS:
if data.get(field):
values[field] = data.pop(field)
for field in self._EXTRA_FIELDS:
if data.get(field):
values[field] = data.pop(field)
description = ""
if values.get("small_description", False):
description = description + "<b>DESCRIPTION:</b><br/>" + values["small_description"]
del values['small_description']
if values.get("access", False):
description = description + "<br/><br/><b>ACCESS:</b><br/>" + values["access"]
del values['access']
if values.get("bug_report", False):
description = description + "<br/><br/><b>BUG REPORT:</b><br/>" + values["bug_report"]
del values['bug_report']
values["description"] = description
values["attachments"] = request.httprequest.files.getlist("attachment")
return values
@http.route(
["/task/create"],
type="http",
auth="public",
methods=['POST'],
website=True,
)
def create_task(self, **kwargs):
# Get form values
user = request.env.user
values = self._compute_form_data(kwargs)
values["project_id"] = user.default_project_id.id
values["partner_id"] = user.partner_id.id
values["user_ids"] = [(6, 0, [user.id])]
files = values.get("attachments", False)
del values['attachments']
# Create task
task_id = request.env["project.task"].sudo().create(values) #use sudo to avoid access error on resource calendar when user_id is set
# Add attachments
for file in files:
attachment_value = {
'name': file.filename,
'datas': base64.encodestring(file.read()),
'res_model': "project.task",
'res_id': task_id,
}
request.env['ir.attachment'].sudo().create(attachment_value)
return request.render("project_task_portal_form.portal_task_created", {})

View File

@@ -0,0 +1 @@
This directory should contain the *.po for Odoo translation.

View File

@@ -0,0 +1,124 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * project_task_portal_form
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 12.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2022-09-22 09:34+0000\n"
"PO-Revision-Date: 2022-09-22 09:34+0000\n"
"Last-Translator: <>\n"
"Language-Team: \n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: \n"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_created
msgid "<br/>\n"
" Task created"
msgstr "<br/>\n"
" Tâche créée"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "<span>Be precise. If there is a bug/error, please describe how to reproduce it</span>"
msgstr "<span>Soyez précis. Si vous avez une erreur, veuillez décrire le scénario pour la reproduire</span>"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "<span>Link toward error or additional information</span>"
msgstr "<span>Lien vers l'erreur ou des informations supplémentaires</span>"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "<span>Paste here the complete error message (ex: error code Odoo)</span>"
msgstr "<span>COpier ici le message d'erreur complet (ex: error code Odoo)</span>"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Access"
msgstr "Accès"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_created
msgid "Back to my Home\n"
" <span class=\"fa fa-long-arrow-right\"/>"
msgstr "Retour sur mon espace\n"
" <span class=\"fa fa-long-arrow-right\"/>"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Bug report"
msgstr "Rapport de bug"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Choose a priority..."
msgstr "Choisir la priorité..."
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Choose a service..."
msgstr "Choisir le logiciel"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Choose a type..."
msgstr "Choisir un type de demande"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Description"
msgstr "Description"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Priority"
msgstr "Priorité"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Request type"
msgstr "Type de demande"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Service"
msgstr "Logiciel"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Submit\n"
" <span class=\"fa fa-long-arrow-right\"/>"
msgstr "Soumettre\n"
" <span class=\"fa fa-long-arrow-right\"/>"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_created
msgid "Task created"
msgstr "Tâche créée"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Task creation form"
msgstr "Formulaire de création de tâche"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_created
msgid "Thanks for request!\n"
" <br/>\n"
" A new task has been saved and you can keep track on your request handling from your portal account!\n"
" <br/>"
msgstr "Merci pour votre demande !\n"
" <br/>\n"
" Une nouvelle tâche a été créée et vous pouvez suivre votre demande depuis votre compte portail !\n"
" <br/>"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Title"
msgstr "Titre"

View File

@@ -0,0 +1,118 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * project_task_portal_form
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 12.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2022-09-22 09:35+0000\n"
"PO-Revision-Date: 2022-09-22 09:35+0000\n"
"Last-Translator: <>\n"
"Language-Team: \n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: \n"
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_created
msgid "<br/>\n"
" Task created"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "<span>Be precise. If there is a bug/error, please describe how to reproduce it</span>"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "<span>Link toward error or additional information</span>"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "<span>Paste here the complete error message (ex: error code Odoo)</span>"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Access"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_created
msgid "Back to my Home\n"
" <span class=\"fa fa-long-arrow-right\"/>"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Bug report"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Choose a priority..."
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Choose a service..."
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Choose a type..."
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Description"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Priority"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Request type"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Service"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Submit\n"
" <span class=\"fa fa-long-arrow-right\"/>"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_created
msgid "Task created"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Task creation form"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_created
msgid "Thanks for request!\n"
" <br/>\n"
" A new task has been saved and you can keep track on your request handling from your portal account!\n"
" <br/>"
msgstr ""
#. module: project_task_portal_form
#: model_terms:ir.ui.view,arch_db:project_task_portal_form.portal_task_creation_form
msgid "Title"
msgstr ""

View File

@@ -0,0 +1,141 @@
<?xml version="1.0" encoding="utf-8"?>
<odoo>
<template id="portal_task_creation_form" name="Portal: Task creation form">
<t t-call="portal.portal_layout">
<!-- <div id="wrap" class="col-lg-6 container"></div> -->
<t t-set="additional_title">Task creation form</t>
<form
action="/task/create"
method="post"
accept-charset="UTF-8"
class="s_website_form col-md-12 mt32"
data-model_name="project.task"
data-force_action=""
data-success_page=""
enctype="multipart/form-data">
<div class="oe_structure" id="oe_structure_portal_task_creation_form_1" />
<input type="hidden" name="csrf_token" t-att-value="request.csrf_token()" />
<div class="row">
<div t-attf-class="form-group #{error.get('name') and 'o_has_error' or ''} col-xl-12">
<label class="col-form-label" for="name">Title</label>
<label class="text-danger"> *</label>
<input type="text" name="name" t-attf-class="form-control #{error.get('name') and 'is-invalid' or ''}" t-att-value="name" required="True" />
</div>
</div>
<div class="row">
<div t-attf-class="form-group #{error.get('request_type_id') and 'o_has_error' or ''} col-xl-6">
<label class="col-form-label" for="request_type_id">Request type</label>
<label class="text-danger"> *</label>
<select name="request_type_id" t-attf-class="form-control" required="True">
<option value="">Choose a type...</option>
<t t-foreach="request_types or []" t-as="request_type">
<option t-att-value="request_type.id">
<t t-esc="request_type.name" />
</option>
</t>
</select>
</div>
<div t-attf-class="form-group #{error.get('service_id') and 'o_has_error' or ''} col-xl-6">
<label class="col-form-label" for="service_id">Service</label>
<label class="text-danger"> *</label>
<select name="service_id" t-attf-class="form-control" required="True">
<option value="">Choose a service...</option>
<t t-foreach="task_services or []" t-as="task_service">
<option t-att-value="task_service.id">
<t t-esc="task_service.name" />
</option>
</t>
</select>
</div>
</div>
<div class="row">
<div t-attf-class="form-group #{error.get('small_description') and 'o_has_error' or ''} col-xl-12">
<label class="col-form-label" for="small_description">Description</label>
<label class="text-danger"> *</label>
<textarea rows="4" name="small_description" t-attf-class="form-control #{error.get('small_description') and 'is-invalid' or ''}" t-att-value="small_description" required="True" />
<span>Be precise. If there is a bug/error, please describe how to reproduce it</span>
</div>
</div>
<div class="row">
<div t-attf-class="form-group #{error.get('access') and 'o_has_error' or ''} col-xl-12">
<label class="col-form-label" for="access">Access</label>
<input type="text" name="access" t-attf-class="form-control #{error.get('access') and 'is-invalid' or ''}" t-att-value="access" />
<span>Link toward error or additional information</span>
</div>
</div>
<div class="row">
<div t-attf-class="form-group #{error.get('bug_report') and 'o_has_error' or ''} col-xl-12">
<label class="col-form-label" for="bug_report">Bug report</label>
<textarea rows="4" name="bug_report" t-attf-class="form-control #{error.get('bug_report') and 'is-invalid' or ''}" t-att-value="bug_report" />
<span>Paste here the complete error message (ex: error code Odoo)</span>
</div>
</div>
<!-- ## IMPORT FILE -->
<div class="form-group form-field form-field-binary" data-model-field="false" data-optional="true">
<label class="col-form-label" for="attachment">
<span>Join document</span>
</label>
<i>
<input type="file" name="attachment" multiple="true" data-show-upload="true" data-show-caption="true" data-show-preview="true" />
</i>
<p class="form-text text-muted">
<i>Join a screenshot to your request or any other helpfull document.</i>
<br />
</p>
</div>
<div class="row">
<div t-attf-class="form-group #{error.get('priority') and 'o_has_error' or ''} col-xl-6">
<label class="col-form-label" for="priority">Priority</label>
<label class="text-danger"> *</label>
<select name="priority" t-attf-class="form-control" required="True">
<option value="">Choose a priority...</option>
<t t-foreach="priorities or []" t-as="priority">
<option t-att-value="priority['id']">
<t t-esc="priority['name']" />
</option>
</t>
</select>
</div>
</div>
<div class="clearfix">
<br />
<button type="submit" class="btn btn-primary float-right mb32 ">
Submit
<span class="fa fa-long-arrow-right" />
</button>
</div>
<div class="oe_structure" id="oe_structure_portal_task_creation_form_2" />
</form>
</t>
</template>
<template id="portal_task_created" name="Portal: Task created">
<t t-call="portal.portal_layout">
<t t-set="additional_title">Task created</t>
<form action="/my/tasks" method="post">
<input type="hidden" name="csrf_token" t-att-value="request.csrf_token()" />
<h3>
<br />
Task created
</h3>
<div>
<p>
Thanks for request!
<br />
A new task has been saved and you can keep track on your request handling from your portal account!
<br />
</p>
</div>
<div class="clearfix">
<br />
<button type="submit" class="btn btn-primary float-right mb32 ">
Back to my Tasks
<span class="fa fa-long-arrow-right" />
</button>
</div>
</form>
</t>
</template>
</odoo>

View File

@@ -1,13 +1,14 @@
============================================
project_only_current_project_tasks_as_parent
============================================
===============
project_timebox
===============
when selecting a parent task in a task, display only tasks from the current project
Add timebox field in tasks to estimate resolution effort.
Installation
============
Use Odoo normal module installation procedure to install ``project_only_current_project_tasks_as_parent``.
Use Odoo normal module installation procedure to install ``project_timebox``.
To configure the timeboxes, go to Project > Configuration > Timeboxes
Known issues / Roadmap
======================
@@ -28,14 +29,16 @@ Credits
Contributors
------------
* Laetitia Da Costa (https://github.com/LaetitiaElabore)
* Stéphan Sainléger (https://github.com/stephansainleger)
* Nicolas Jeudy (https://github.com/njeudy)
Funders
-------
The development of this module has been financially supported by:
* Elabore (https://elabore.coop)
* Mycéliandre (https://myceliandre.fr)
* Lokavaluto (https://lokavaluto.fr)
Maintainer

View File

@@ -0,0 +1 @@
from . import models

View File

@@ -0,0 +1,93 @@
# Copyright 2022 Stéphan Sainléger (Elabore)
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
{
"name": "project_timebox",
"version": "16.0.1.0.0",
"author": "Elabore",
"website": "https://elabore.coop",
"maintainer": "Stéphan Sainléger",
"license": "AGPL-3",
"category": "Project",
"summary": "Add timebox field in tasks to estimate resolution effort.",
"description": """
:image: https://img.shields.io/badge/licence-AGPL--3-blue.svg
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
:alt: License: AGPL-3
===============
project_timebox
===============
Add timebox field in tasks to estimate resolution effort.
Installation
============
Use Odoo normal module installation procedure to install ``project_timebox``.
To configure the timeboxes, go to Project > Configuration > Timeboxes
Known issues / Roadmap
======================
None yet.
Bug Tracker
===========
Bugs are tracked on `our issues website <https://github.com/elabore-coop/project-tools/issues>`_. In case of
trouble, please check there if your issue has already been
reported. If you spotted it first, help us smashing it by providing a
detailed and welcomed feedback.
Credits
=======
Contributors
------------
* Stéphan Sainléger (https://github.com/stephansainleger)
* Nicolas Jeudy (https://github.com/njeudy)
Funders
-------
The development of this module has been financially supported by:
* Elabore (https://elabore.coop)
* Mycéliandre (https://myceliandre.fr)
* Lokavaluto (https://lokavaluto.fr)
Maintainer
----------
This module is maintained by Elabore.
""",
# any module necessary for this one to work correctly
"depends": [
"base",
"project",
],
"qweb": [
# "static/src/xml/*.xml",
],
"external_dependencies": {
"python": [],
},
# always loaded
"data": [
"security/ir.model.access.csv",
"views/project_task.xml",
"views/timebox.xml",
"views/portal_template.xml",
],
# only loaded in demonstration mode
"demo": [],
"js": [],
"css": [],
"installable": True,
# Install this module automatically if all dependency have been previously
# and independently installed. Used for synergetic or glue modules.
"auto_install": False,
"application": False,
}

View File

@@ -0,0 +1 @@
This directory should contain the *.po for Odoo translation.

View File

@@ -0,0 +1,91 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * project_timebox
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 14.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2022-08-01 09:34+0000\n"
"PO-Revision-Date: 2022-08-01 09:34+0000\n"
"Last-Translator: \n"
"Language-Team: \n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: \n"
#. module: project_timebox
#: model:ir.model.fields,field_description:project_timebox.field_timebox__create_uid
msgid "Created by"
msgstr "Créé par"
#. module: project_timebox
#: model:ir.model.fields,field_description:project_timebox.field_timebox__create_date
msgid "Created on"
msgstr "Créé le"
#. module: project_timebox
#: model:ir.model.fields,field_description:project_timebox.field_project_task__display_name
#: model:ir.model.fields,field_description:project_timebox.field_timebox__display_name
msgid "Display Name"
msgstr "Nom affiché"
#. module: project_timebox
#: model:ir.model.fields,field_description:project_timebox.field_project_task__id
#: model:ir.model.fields,field_description:project_timebox.field_timebox__id
msgid "ID"
msgstr ""
#. module: project_timebox
#: model:ir.model.fields,field_description:project_timebox.field_project_task____last_update
#: model:ir.model.fields,field_description:project_timebox.field_timebox____last_update
msgid "Last Modified on"
msgstr "Dernière modification le"
#. module: project_timebox
#: model:ir.model.fields,field_description:project_timebox.field_timebox__write_uid
msgid "Last Updated by"
msgstr "Dernière mise à jour par"
#. module: project_timebox
#: model:ir.model.fields,field_description:project_timebox.field_timebox__write_date
msgid "Last Updated on"
msgstr "Dernière mise à jour le"
#. module: project_timebox
#: model:ir.model.fields,field_description:project_timebox.field_timebox__sequence
msgid "Sequence"
msgstr "Séquence"
#. module: project_timebox
#: model:ir.model,name:project_timebox.model_project_task
msgid "Task"
msgstr "Tâche"
#. module: project_timebox
#: model:ir.model,name:project_timebox.model_timebox
msgid "Timebox"
msgstr "Estimation"
#. module: project_timebox
#: model:ir.model.fields,field_description:project_timebox.field_project_task__timebox_max_id
msgid "Timebox Max"
msgstr "Estimation haute"
#. module: project_timebox
#: model:ir.model.fields,field_description:project_timebox.field_project_task__timebox_min_id
msgid "Timebox Min"
msgstr "Estimation basse"
#. module: project_timebox
#: model:ir.actions.act_window,name:project_timebox.act_timeboxes_list
#: model:ir.ui.menu,name:project_timebox.menu_timeboxes
#: model_terms:ir.ui.view,arch_db:project_timebox.timebox_view_tree
msgid "Timeboxes"
msgstr "Estimations"
#. module: project_timebox
#: model:ir.model.fields,field_description:project_timebox.field_timebox__name
msgid "name"
msgstr "nom"

View File

@@ -0,0 +1,2 @@
from . import timebox
from . import project_task

View File

@@ -0,0 +1,9 @@
from odoo import models, fields
class Task(models.Model):
_inherit = "project.task"
timebox_min_id = fields.Many2one('timebox', string='Timebox Min')
timebox_max_id = fields.Many2one('timebox', string='Timebox Max')

View File

@@ -0,0 +1,9 @@
from odoo import models, fields
class Task(models.Model):
_name = "timebox"
_description = "Timebox"
name = fields.Char('name', required=True)
sequence = fields.Integer()

View File

@@ -0,0 +1,3 @@
id,name,model_id/id,group_id/id,perm_read,perm_write,perm_create,perm_unlink
access_timebox_user,project.timebox.user,model_timebox,,1,0,0,0
access_timebox_manager,project.timebox.manager,model_timebox,project.group_project_manager,1,1,1,1
1 id name model_id/id group_id/id perm_read perm_write perm_create perm_unlink
2 access_timebox_user project.timebox.user model_timebox 1 0 0 0
3 access_timebox_manager project.timebox.manager model_timebox project.group_project_manager 1 1 1 1

View File

@@ -0,0 +1,17 @@
<?xml version="1.0" encoding="utf-8"?>
<odoo>
<template id="portal_my_task_timebox" name="My Task: Timebox" inherit_id="project.portal_my_task" priority="40">
<xpath expr="//div[@t-if='task.date_deadline']" position="after">
<div id="timebox" class="row mb-2">
<div t-if="task.timebox_min_id">
<strong>Timebox Min:</strong>
<span t-field="task.timebox_min_id" />
</div>
<div t-if="task.timebox_max_id">
<strong>Timebox Max:</strong>
<span t-field="task.timebox_max_id" />
</div>
</div>
</xpath>
</template>
</odoo>

View File

@@ -0,0 +1,15 @@
<?xml version="1.0" encoding="UTF-8"?>
<odoo>
<record id="view_task_form2_timebox" model="ir.ui.view">
<field name="name">project.task.form.timebox</field>
<field name="model">project.task</field>
<field name="inherit_id" ref="project.view_task_form2" />
<field name="priority" eval="99" />
<field name="arch" type="xml">
<xpath expr="//field[@name='date_deadline']" position="after">
<field name="timebox_min_id" widget="selection" />
<field name="timebox_max_id" widget="selection" />
</xpath>
</field>
</record>
</odoo>

View File

@@ -0,0 +1,23 @@
<?xml version="1.0" encoding="utf-8"?>
<odoo>
<record id="timebox_view_tree" model="ir.ui.view">
<field name="name">timebox.view.tree</field>
<field name="model">timebox</field>
<field name="arch" type="xml">
<tree string="Timeboxes" editable="top" default_order="sequence">
<field name="sequence" widget="handle" />
<field name="name" />
</tree>
</field>
</record>
<record id="act_timeboxes_list" model="ir.actions.act_window">
<field name="name">Timeboxes</field>
<field name="res_model">timebox</field>
<field name="view_mode">tree</field>
</record>
<menuitem id="menu_timeboxes" action="act_timeboxes_list" parent="project.menu_project_config" sequence="99" name="Timeboxes" />
</odoo>

View File

@@ -0,0 +1,43 @@
======================
project_usability_misc
======================
Miscellaneous usability improvements for project management.
- Add a "All tasks" button in "Tasks" Menu to display all tasks.
Installation
============
Use Odoo normal module installation procedure to install
``project_usability_misc``.
Known issues / Roadmap
======================
None yet.
Bug Tracker
===========
Bugs are tracked on `our issues website <https://github.com/elabore-coop/project-tools/issues>`_. In case of
trouble, please check there if your issue has already been
reported. If you spotted it first, help us smashing it by providing a
detailed and welcomed feedback.
Credits
=======
Contributors
------------
* Boris Gallet
Funders
-------
The development of this module has been financially supported by:
* Elabore (https://elabore.coop)
Maintainer
----------
This module is maintained by Elabore.

View File

@@ -0,0 +1 @@
# Empty __init__.py file for the project_usability_misc module.

View File

@@ -0,0 +1,23 @@
# Copyright 2025 Boris Gallet (Elabore)
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
{
"name": "project_usability_misc",
"version": "16.0.1.0.0",
"author": "Elabore",
"website": "https://github.com/elabore-coop/project-tools",
"maintainer": "boris.gallet@elabore.coop",
"license": "AGPL-3",
"category": "Tools",
"summary": "Miscellaneous usability improvements for project management",
"depends": [
"base",
"project",
],
"data": [
"views/project_task.xml",
],
"demo": [],
"installable": True,
"auto_install": False,
"application": False,
}

Some files were not shown because too many files have changed in this diff Show More