Browse Source

Jun 14: [ADD] Initial Commit 'odoo_pipedrive_connector'

pull/320/head
RisvanaCybro 11 months ago
parent
commit
81a3c99438
  1. 52
      odoo_pipedrive_connector/README.rst
  2. 23
      odoo_pipedrive_connector/__init__.py
  3. 46
      odoo_pipedrive_connector/__manifest__.py
  4. 22
      odoo_pipedrive_connector/controllers/__init__.py
  5. 231
      odoo_pipedrive_connector/controllers/odoo_pipedrive_connector.py
  6. 6
      odoo_pipedrive_connector/doc/RELEASE_NOTES.md
  7. 27
      odoo_pipedrive_connector/models/__init__.py
  8. 129
      odoo_pipedrive_connector/models/crm_lead.py
  9. 38
      odoo_pipedrive_connector/models/pipedrive_record.py
  10. 75
      odoo_pipedrive_connector/models/product_category.py
  11. 162
      odoo_pipedrive_connector/models/product_template.py
  12. 434
      odoo_pipedrive_connector/models/res_company.py
  13. 113
      odoo_pipedrive_connector/models/res_partner.py
  14. 2
      odoo_pipedrive_connector/security/ir.model.access.csv
  15. BIN
      odoo_pipedrive_connector/static/description/assets/icons/check.png
  16. BIN
      odoo_pipedrive_connector/static/description/assets/icons/chevron.png
  17. BIN
      odoo_pipedrive_connector/static/description/assets/icons/cogs.png
  18. BIN
      odoo_pipedrive_connector/static/description/assets/icons/consultation.png
  19. BIN
      odoo_pipedrive_connector/static/description/assets/icons/ecom-black.png
  20. BIN
      odoo_pipedrive_connector/static/description/assets/icons/education-black.png
  21. BIN
      odoo_pipedrive_connector/static/description/assets/icons/hotel-black.png
  22. BIN
      odoo_pipedrive_connector/static/description/assets/icons/license.png
  23. BIN
      odoo_pipedrive_connector/static/description/assets/icons/lifebuoy.png
  24. BIN
      odoo_pipedrive_connector/static/description/assets/icons/logo.png
  25. BIN
      odoo_pipedrive_connector/static/description/assets/icons/manufacturing-black.png
  26. BIN
      odoo_pipedrive_connector/static/description/assets/icons/pos-black.png
  27. BIN
      odoo_pipedrive_connector/static/description/assets/icons/puzzle.png
  28. BIN
      odoo_pipedrive_connector/static/description/assets/icons/restaurant-black.png
  29. BIN
      odoo_pipedrive_connector/static/description/assets/icons/service-black.png
  30. BIN
      odoo_pipedrive_connector/static/description/assets/icons/trading-black.png
  31. BIN
      odoo_pipedrive_connector/static/description/assets/icons/training.png
  32. BIN
      odoo_pipedrive_connector/static/description/assets/icons/update.png
  33. BIN
      odoo_pipedrive_connector/static/description/assets/icons/user.png
  34. BIN
      odoo_pipedrive_connector/static/description/assets/icons/wrench.png
  35. BIN
      odoo_pipedrive_connector/static/description/assets/misc/categories.png
  36. BIN
      odoo_pipedrive_connector/static/description/assets/misc/check-box.png
  37. BIN
      odoo_pipedrive_connector/static/description/assets/misc/compass.png
  38. BIN
      odoo_pipedrive_connector/static/description/assets/misc/corporate.png
  39. BIN
      odoo_pipedrive_connector/static/description/assets/misc/customer-support.png
  40. BIN
      odoo_pipedrive_connector/static/description/assets/misc/cybrosys-logo.png
  41. BIN
      odoo_pipedrive_connector/static/description/assets/misc/features.png
  42. BIN
      odoo_pipedrive_connector/static/description/assets/misc/logo.png
  43. BIN
      odoo_pipedrive_connector/static/description/assets/misc/pictures.png
  44. BIN
      odoo_pipedrive_connector/static/description/assets/misc/pie-chart.png
  45. BIN
      odoo_pipedrive_connector/static/description/assets/misc/right-arrow.png
  46. BIN
      odoo_pipedrive_connector/static/description/assets/misc/star.png
  47. BIN
      odoo_pipedrive_connector/static/description/assets/misc/support.png
  48. BIN
      odoo_pipedrive_connector/static/description/assets/misc/whatsapp.png
  49. BIN
      odoo_pipedrive_connector/static/description/assets/modules/agriculture.png
  50. BIN
      odoo_pipedrive_connector/static/description/assets/modules/hospital.png
  51. BIN
      odoo_pipedrive_connector/static/description/assets/modules/l1.png
  52. BIN
      odoo_pipedrive_connector/static/description/assets/modules/l2.png
  53. BIN
      odoo_pipedrive_connector/static/description/assets/modules/l3.png
  54. BIN
      odoo_pipedrive_connector/static/description/assets/modules/l4.png
  55. BIN
      odoo_pipedrive_connector/static/description/assets/modules/l5.png
  56. BIN
      odoo_pipedrive_connector/static/description/assets/modules/l6.png
  57. BIN
      odoo_pipedrive_connector/static/description/assets/modules/project_dashboard.png
  58. BIN
      odoo_pipedrive_connector/static/description/assets/modules/shopify.png
  59. BIN
      odoo_pipedrive_connector/static/description/assets/modules/systray.png
  60. BIN
      odoo_pipedrive_connector/static/description/assets/modules/woocomp_connector.png
  61. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/1.png
  62. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/2.png
  63. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot.png
  64. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot0.png
  65. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot1.png
  66. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot10.png
  67. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot11.png
  68. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot12.png
  69. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot13.png
  70. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot14.png
  71. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot15.png
  72. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot16.png
  73. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot17.png
  74. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot18.png
  75. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot19.png
  76. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot2.png
  77. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot20.png
  78. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot21.png
  79. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot22.png
  80. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot23.png
  81. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot24.png
  82. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot25.png
  83. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot3.png
  84. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot4.png
  85. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot6.png
  86. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot7.png
  87. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot8.png
  88. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot9.png
  89. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/hero.gif
  90. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/sync_contact.png
  91. BIN
      odoo_pipedrive_connector/static/description/assets/screenshots/sync_lead.png
  92. BIN
      odoo_pipedrive_connector/static/description/banner.jpg
  93. BIN
      odoo_pipedrive_connector/static/description/icon.png
  94. 795
      odoo_pipedrive_connector/static/description/index.html
  95. 47
      odoo_pipedrive_connector/views/res_company_views.xml

52
odoo_pipedrive_connector/README.rst

@ -0,0 +1,52 @@
.. image:: https://img.shields.io/badge/license-AGPL--3-blue.svg
:target: https://www.gnu.org/licenses/agpl-3.0-standalone.html
:alt: License: AGPL-3
Odoo Pipedrive Connector
========================
Integrate Products, Contacts and Leads between Pipedrive and Odoo
Configuration
=============
Please note that this feature is only supported on secure (HTTPS) URLs.
* First you need to create Pipedrive Token.
1) Go to https://demo-sandbox4.pipedrive.com/settings/api and login to your account.
2) Click on your profile picture at the right corner.
3) Click on Personal preferences.
4) Under API, you can see the Pipedrive token.
License
-------
GNU AFFERO GENERAL PUBLIC LICENSE, Version 3 (AGPLv3)
(https://www.gnu.org/licenses/agpl-3.0-standalone.html)
Company
-------
* `Cybrosys Techno Solutions <https://cybrosys.com/>`__
Credits
-------
* Developer: (V16) Unnimaya C O, Contact: odoo@cybrosys.com
Contacts
--------
* Mail Contact : odoo@cybrosys.com
* Website : https://cybrosys.com
Bug Tracker
-----------
Bugs are tracked on GitHub Issues. In case of trouble, please check there if your issue has already been reported.
Maintainer
==========
.. image:: https://cybrosys.com/images/logo.png
:target: https://cybrosys.com
This module is maintained by Cybrosys Technologies.
For support and more information, please visit `Our Website <https://cybrosys.com/>`__
Further information
===================
HTML Description: `<static/description/index.html>`__

23
odoo_pipedrive_connector/__init__.py

@ -0,0 +1,23 @@
# -*- coding: utf-8 -*-
################################################################################
#
# Cybrosys Technologies Pvt. Ltd.
#
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>).
# Author: Unnimaya C O (odoo@cybrosys.com)
#
# You can modify it under the terms of the GNU AFFERO
# GENERAL PUBLIC LICENSE (AGPL v3), Version 3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details.
#
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
# (AGPL v3) along with this program.
# If not, see <http://www.gnu.org/licenses/>.
#
################################################################################
from . import controllers
from . import models

46
odoo_pipedrive_connector/__manifest__.py

@ -0,0 +1,46 @@
# -*- coding: utf-8 -*-
################################################################################
#
# Cybrosys Technologies Pvt. Ltd.
#
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>).
# Author: Unnimaya C O (odoo@cybrosys.com)
#
# You can modify it under the terms of the GNU AFFERO
# GENERAL PUBLIC LICENSE (AGPL v3), Version 3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details.
#
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
# (AGPL v3) along with this program.
# If not, see <http://www.gnu.org/licenses/>.
#
################################################################################
{
'name': "Odoo Pipedrive Connector",
'version': '16.0.1.0.0',
'category': 'Productivity',
'summary': """Integrate contacts, products and leads between Pipedrive
and Odoo""",
'description': """This module helps to successfully import all products,
contacts and leads between Pipedrive to Odoo. Also, it is possible to
export all these data from Odoo to Pipedrive. All import and export
operations can be performed in a single button click.""",
'author': 'Cybrosys Techno Solutions',
'company': 'Cybrosys Techno Solutions',
'maintainer': 'Cybrosys Techno Solutions',
'website': 'https://www.cybrosys.com',
'depends': ['account', 'crm'],
'data': [
'security/ir.model.access.csv',
'views/res_company_views.xml'
],
'images': ['static/description/banner.jpg'],
'license': 'AGPL-3',
'installable': True,
'auto_install': False,
'application': False,
}

22
odoo_pipedrive_connector/controllers/__init__.py

@ -0,0 +1,22 @@
# -*- coding: utf-8 -*-
################################################################################
#
# Cybrosys Technologies Pvt. Ltd.
#
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>).
# Author: Unnimaya C O (odoo@cybrosys.com)
#
# You can modify it under the terms of the GNU AFFERO
# GENERAL PUBLIC LICENSE (AGPL v3), Version 3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details.
#
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
# (AGPL v3) along with this program.
# If not, see <http://www.gnu.org/licenses/>.
#
################################################################################
from . import odoo_pipedrive_connector

231
odoo_pipedrive_connector/controllers/odoo_pipedrive_connector.py

@ -0,0 +1,231 @@
# -*- coding: utf-8 -*-
################################################################################
#
# Cybrosys Technologies Pvt. Ltd.
#
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>).
# Author: Unnimaya C O (odoo@cybrosys.com)
#
# You can modify it under the terms of the GNU AFFERO
# GENERAL PUBLIC LICENSE (AGPL v3), Version 3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details.
#
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
# (AGPL v3) along with this program.
# If not, see <http://www.gnu.org/licenses/>.
#
################################################################################
import json
from odoo import fields, http
from odoo.http import request
class PipedriveWebhook(http.Controller):
""" This controller is responsible for receiving Webhooks from Pipedrive"""
@http.route('/update_pipedrive_product', type="json", auth="public", csrf=False,
methods=['POST'])
def get_updated_product_details(self, **kw):
"""Webhook for receiving the updated product details."""
data = json.loads(request.httprequest.data.decode('utf-8'))
request.env.company.sudo().create_product_category()
uom_id = 1
if data['current']['unit']:
for rec in request.env['uom.uom'].sudo().search([]).mapped(
'name'):
if rec.lower() == data['current']['unit'].lower():
uom_id = request.env['uom.uom'].sudo().search(
[('name', '=', rec)]).id
pipedrive_product = request.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'product'),
('pipedrive_reference', '=', data['current']['id'])])
product_template = request.env['product.template'].sudo().browse(
pipedrive_product.odoo_ref)
pipedrive_categ = request.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'categ'), ('pipedrive_reference', '=',
data['current']['category'])])
category_id = request.env['product.category'].sudo().browse(
pipedrive_categ.odoo_ref) if pipedrive_categ else 1
currency = request.env['res.currency']
if data['current']['prices']:
currency = request.env['res.currency'].sudo().search(
[('name', '=', data['current']['prices'][0]['currency']),
('active', 'in', [True, False])])
if not currency.active:
currency.active = True
list_price = currency._convert(
data['current']['prices'][0]['price'],
request.env.company.currency_id, request.env.company,
fields.date.today()) if data['current']['prices'][0][
'price'] else 0.0,
standard_price = currency._convert(
data['current']['prices'][0]['cost'],
request.env.company.currency_id, request.env.company,
fields.date.today()) if data['current']['prices'][0][
'cost'] else 0.0,
if product_template:
update_values = {}
if product_template.name != data['current']['name']:
update_values['name'] = data['current']['name']
if product_template.description != data['current']['description']:
update_values['description'] = data['current']['description']
if product_template.uom_id.id != uom_id:
update_values['uom_id'] = uom_id
update_values['uom_po_id'] = uom_id
if product_template.active != data['current']['active_flag']:
update_values['active'] = data['current']['active_flag']
if product_template.standard_price != standard_price[0]:
update_values['standard_price'] = standard_price[0]
if product_template.list_price != list_price[0]:
update_values['list_price'] = list_price[0]
if product_template.categ_id.id != category_id:
update_values['categ_id'] = category_id
product_template.update_from_pipedrive = False
if update_values:
product_template['update_from_pipedrive'] = True
product_template.sudo().write(update_values)
@http.route('/add_pipedrive_product', type="json", auth="public",
csrf=False, methods=['POST'])
def get_added_product_details(self, **kw):
"""Webhook for receiving the new product details."""
if json.loads(request.httprequest.data.decode('utf-8'))[
'meta']['change_source'] != 'api':
added_data = json.loads(request.httprequest.data.decode('utf-8'))[
'current']
request.env.company.sudo().create_product_category()
uom = 1
currency = request.env['res.currency']
if added_data['prices']:
currency = request.env['res.currency'].sudo().search(
[('name', '=', added_data['prices'][0]['currency']),
('active', 'in', [True, False])])
if not currency.active:
currency.active = True
if added_data['unit']:
for rec in request.env['uom.uom'].sudo().search([]).mapped(
'name'):
if rec.lower() == added_data['unit'].lower():
uom = request.env['uom.uom'].sudo().search(
[('name', '=', rec)]).id
if not request.env['pipedrive.record'].sudo().search(
[('pipedrive_reference', '=', added_data['id']),
('record_type', '=', 'product')]):
product = request.env['product.template'].sudo().create({
'name': added_data['name'],
'description': added_data['description'],
'uom_id': uom,
'default_code': added_data['code'],
'uom_po_id': uom,
'taxes_id': False,
'list_price': currency._convert(
added_data['prices'][0]['price'],
request.env.company.currency_id, request.env.company,
fields.date.today()) if
added_data['prices'][0][
'price'] else 0.0,
'standard_price': currency._convert(
added_data['prices'][0]['cost'],
request.env.company.currency_id, request.env.company,
fields.date.today()) if
added_data['prices'][0][
'cost'] else 0.0,
'categ_id': request.env['pipedrive.record'].sudo().search(
[(
'pipedrive_reference', '=',
added_data['category'])])[0].odoo_ref if request.env['pipedrive.record'].sudo().search(
[(
'pipedrive_reference', '=',
added_data['category'])]) else 1
if added_data['category'] else 1,
'pipedrive_reference': added_data['id']
})
request.env['pipedrive.record'].sudo().create({
'pipedrive_reference': added_data['id'],
'record_type': 'product',
'odoo_ref': product.id
})
product.taxes_id.unlink()
if added_data['tax'] != 0:
tax = request.env['account.tax'].sudo().search(
[('amount_type', '=', 'percent'),
('type_tax_use', '=', 'sale'), ('amount',
'=',
added_data['tax'])],limit=1)
if not tax:
tax = request.env['account.tax'].sudo().create({
'name': 'Tax ' + str(added_data['tax']) + '%',
'amount_type': 'percent',
'type_tax_use': 'sale',
'amount': added_data['tax']
})
product.sudo().write({
"taxes_id": [(4, tax.id)]
})
@http.route('/delete_pipedrive_product', type="json", auth="public",
csrf=False, methods=['POST'])
def get_deleted_product_details(self, **kw):
"""Webhook for receiving the deleted product details."""
deleted_data = json.loads(request.httprequest.data.decode('utf-8'))
pipedrive_product = request.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'product'), ('pipedrive_reference', '=', deleted_data['meta']['id'])])
request.env['product.template'].sudo().browse(pipedrive_product.odoo_ref).sudo().write(
{'active': False})
pipedrive_product.sudo().write(
{'active': False})
@http.route('/update_pipedrive_contact', type="json", auth="public", csrf=False,
methods=['POST'])
def get_updated_contact_details(self, **kw):
"""Webhook for receiving the updated contact details."""
data = json.loads(request.httprequest.data.decode('utf-8'))
pipedrive_contact = request.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'partner'),
('pipedrive_reference', '=', data['current']['id'])])
partner = request.env['res.partner'].sudo().browse([(
pipedrive_contact.odoo_ref)])
if partner:
partner.sudo().write({
'name': data['current']['name'],
'email': data['current']['email'][0]['value'],
'phone': data['current']['phone'][0]['value'],
})
@http.route('/add_pipedrive_contact', type="json", auth="public", csrf=False,
methods=['POST'])
def get_added_contact_details(self, **kw):
"""Webhook for receiving the added contact details."""
if json.loads(request.httprequest.data.decode('utf-8'))['meta']['change_source'] != 'api':
data = json.loads(request.httprequest.data.decode('utf-8'))['current']
if not request.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'partner'), ('pipedrive_reference', '=', data['id'])]):
person = request.env['res.partner'].sudo().create({
'name': data['name'],
'phone': data['phone'][0]['value'],
'email': data['email'][0]['value'],
'pipedrive_reference': data['id']
})
request.env['pipedrive.record'].sudo().create({
'pipedrive_reference': data['id'],
'record_type': 'partner',
'odoo_ref': person.id
})
@http.route('/delete_pipedrive_contact', type="json", auth="public",
methods=['POST'], csrf=False)
def get_deleted_contact_details(self, **kw):
"""Webhook for receiving the deleted contact details."""
data = json.loads(request.httprequest.data.decode('utf-8'))
pipedrive_contact = request.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'partner'),
('pipedrive_reference', '=', data['meta']['id'])])
request.env['product.template'].sudo().browse(pipedrive_contact.odoo_ref).sudo().write(
{'active': False})
pipedrive_contact.sudo().write(
{'active': False})

6
odoo_pipedrive_connector/doc/RELEASE_NOTES.md

@ -0,0 +1,6 @@
## Module <odoo_pipedrive_connector>
#### 23.05.2024
#### Version 16.0.1.0.0
#### ADD
- Initial commit for Odoo Pipedrive Connector

27
odoo_pipedrive_connector/models/__init__.py

@ -0,0 +1,27 @@
# -*- coding: utf-8 -*-
################################################################################
#
# Cybrosys Technologies Pvt. Ltd.
#
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>).
# Author: Unnimaya C O (odoo@cybrosys.com)
#
# You can modify it under the terms of the GNU AFFERO
# GENERAL PUBLIC LICENSE (AGPL v3), Version 3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details.
#
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
# (AGPL v3) along with this program.
# If not, see <http://www.gnu.org/licenses/>.
#
################################################################################
from . import crm_lead
from . import pipedrive_record
from . import product_category
from . import product_template
from . import res_company
from . import res_partner

129
odoo_pipedrive_connector/models/crm_lead.py

@ -0,0 +1,129 @@
# -*- coding: utf-8 -*-
################################################################################
#
# Cybrosys Technologies Pvt. Ltd.
#
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>).
# Author: Unnimaya C O (odoo@cybrosys.com)
#
# You can modify it under the terms of the GNU AFFERO
# GENERAL PUBLIC LICENSE (AGPL v3), Version 3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details.
#
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
# (AGPL v3) along with this program.
# If not, see <http://www.gnu.org/licenses/>.
#
################################################################################
import json
import requests
from odoo import api, fields, models
from odoo.exceptions import ValidationError
class CrmLead(models.Model):
"""Inherits crm Lead for including Pipedrive fields and functions"""
_inherit = 'crm.lead'
pipedrive_reference = fields.Char(string='Pipedrive Id',
help="Pipedrive reference of the lead")
def write(self, vals):
"""Inherited to add the code for updating the product details in
Pipedrive"""
data = {}
if 'name' in vals.keys():
data['title'] = vals['name']
if 'partner_id' in vals.keys():
pipedrive_reference = self.env['res.partner'].browse(
vals['partner_id']).pipedrive_reference
if not pipedrive_reference:
pipedrive_reference = (
self.env.user.company_id.sudo().create_contact(
self.env['res.partner'].sudo().browse(
vals['partner_id'])))
data['person_id'] = int(pipedrive_reference)
if 'expected_revenue' in vals.keys():
data['value'] = {
'amount': vals['expected_revenue'],
'currency': self.env.company.currency_id.name
}
for rec in self:
pipedrive_lead = self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'lead'), ('odoo_ref', '=', rec.id)])
if pipedrive_lead and data:
headers = {
'Content-Type': 'application/json',
'Accept': 'application/json'
}
response = requests.patch(
url=f'https://api.pipedrive.com/v1/leads/'
f'{rec.pipedrive_reference}',
params={
'api_token': rec.env.user.company_id.api_key,
}, timeout=10, headers=headers, data=json.dumps(data))
if 'error' in response.json().keys():
raise ValidationError(
response.json()['error'])
return super().write(vals)
def unlink(self):
"""Inherited to add the code for deleting the product from Pipedrive"""
for rec in self:
pipedrive_lead = self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'lead'), ('odoo_ref', '=', rec.id)])
if pipedrive_lead:
headers = {
'Content-Type': 'application/json',
'Accept': 'application/json'
}
response = requests.delete(
url=f'https://api.pipedrive.com/v1/leads/'
f'{pipedrive_lead.pipedrive_reference}',
params={
'api_token': self.env.user.company_id.api_key,
}, timeout=10, headers=headers)
if 'error' in response.json().keys():
raise ValidationError(
response.json()['error'])
return super().unlink()
@api.model_create_multi
def create(self, vals_list):
"""Inherited to add the lead to pipedrive"""
res = super().create(vals_list)
if self.env.company.lead_synced and not self.env['pipedrive.record'].search(
[('record_type', '=', 'lead',), ('odoo_ref', '=', res.id)]):
for lead in vals_list:
if lead['partner_id']:
data = {
'title': lead['name'],
'person_id': int(self.env['pipedrive.record'].search(
[('record_type', '=', 'partner',),
('odoo_ref', '=', lead['partner_id'])]).pipedrive_reference),
'value': {
'amount': lead[
'expected_revenue'] if 'expected_revenue'
in lead.keys() else 0.0,
'currency': self.env.company.currency_id.name
}
}
response = requests.post(
url='https://api.pipedrive.com/v1/leads',
params={
'api_token': self.env.company.api_key,
}, json=data, timeout=10)
if not response.json()['success']:
raise ValidationError(
response.json()['error'] + '. ' + response.json()[
'error_info'])
self.env['pipedrive.record'].sudo().create({
'pipedrive_reference': response.json()['data']['id'],
'record_type': 'lead',
'odoo_ref': res.id
})
return res

38
odoo_pipedrive_connector/models/pipedrive_record.py

@ -0,0 +1,38 @@
# -*- coding: utf-8 -*-
################################################################################
#
# Cybrosys Technologies Pvt. Ltd.
#
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>).
# Author: Unnimaya C O (odoo@cybrosys.com)
#
# You can modify it under the terms of the GNU AFFERO
# GENERAL PUBLIC LICENSE (AGPL v3), Version 3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details.
#
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
# (AGPL v3) along with this program.
# If not, see <http://www.gnu.org/licenses/>.
#
################################################################################
from odoo import fields, models
class PipedriveRecord(models.Model):
"""Model to hold the Pipedrive Records"""
_name = 'pipedrive.record'
_description = 'Pipedrive Record'
_rec_name = 'pipedrive_reference'
pipedrive_reference = fields.Char(string='Pipedrive Id',
help="Pipedrive reference of the record")
record_type = fields.Selection(
[('product', 'Product'), ('lead', 'Lead'), ('partner', 'Partner'), ('categ', 'Category')],
string='Type',
help='Type of record')
odoo_ref = fields.Integer(string='Odoo Reference',
help="Odoo reference of the record")

75
odoo_pipedrive_connector/models/product_category.py

@ -0,0 +1,75 @@
# -*- coding: utf-8 -*-
################################################################################
#
# Cybrosys Technologies Pvt. Ltd.
#
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>).
# Author: Unnimaya C O (odoo@cybrosys.com)
#
# You can modify it under the terms of the GNU AFFERO
# GENERAL PUBLIC LICENSE (AGPL v3), Version 3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details.
#
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
# (AGPL v3) along with this program.
# If not, see <http://www.gnu.org/licenses/>.
#
################################################################################
import json
import requests
from odoo import fields, models
from odoo.exceptions import ValidationError
class ProductCategory(models.Model):
"""Inherits product_category for including Pipedrive fields and functions"""
_inherit = 'product.category'
pipedrive_reference = fields.Char(string='Pipedrive Id',
help="Pipedrive Id of the Partner")
def write(self, vals):
"""Inherited to update product field in pipedrive"""
data = {}
pipedrive_categ = self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'categ'), ('odoo_ref', '=', self.id)])
if 'name' in vals.keys() and pipedrive_categ:
data['label'] = vals['name']
headers = {
'Content-Type': 'application/json',
'Accept': 'application/json'
}
response = requests.put(
url=f'https://api.pipedrive.com/v1/productFields/'
f'{self.pipedrive_reference}',
params={
'api_token': self.env.user.company_id.api_key,
}, timeout=10, headers=headers, data=json.dumps(data))
if 'error' in response.json().keys():
raise ValidationError(
response.json()['error'])
return super().write(vals)
def unlink(self):
"""Inherited to delete the product field from Pipedrive"""
pipedrive_categ = self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'categ'), ('odoo_ref', '=', self.id)])
if pipedrive_categ:
headers = {
'Content-Type': 'application/json',
'Accept': 'application/json'
}
response = requests.delete(
url=f'https://api.pipedrive.com/v1/productFields/'
f'{self.pipedrive_reference}',
params={
'api_token': self.env.user.company_id.api_key,
}, timeout=10, headers=headers)
if 'error' in response.json().keys():
raise ValidationError(
response.json()['error'])
return super().unlink()

162
odoo_pipedrive_connector/models/product_template.py

@ -0,0 +1,162 @@
# -*- coding: utf-8 -*-
################################################################################
#
# Cybrosys Technologies Pvt. Ltd.
#
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>).
# Author: Unnimaya C O (odoo@cybrosys.com)
#
# You can modify it under the terms of the GNU AFFERO
# GENERAL PUBLIC LICENSE (AGPL v3), Version 3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details.
#
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
# (AGPL v3) along with this program.
# If not, see <http://www.gnu.org/licenses/>.
#
################################################################################
import json
import requests
from odoo import api, fields, models
from odoo.exceptions import ValidationError
class ProductTemplate(models.Model):
"""Inherits product template for including Pipedrive fields ab=nd
functions"""
_inherit = 'product.template'
pipedrive_reference = fields.Char(string='Pipedrive Id',
help="Pipedrive Id of the Product")
update_from_pipedrive = fields.Boolean(string='Update from Pipedrive',
help="True if the update is from pipedrive")
def write(self, vals):
"""Inherited to update the Pipedrive product."""
data = {}
if 'name' in vals.keys():
data['name'] = vals['name']
if 'uom_id' in vals.keys():
data['unit'] = self.env['uom.uom'].browse(vals['uom_id']).name
if 'taxes_id' in vals.keys():
if vals['taxes_id'] and type(vals['taxes_id'][0]) == list:
total_tax = 0.0
for tax in self.env['account.tax'].sudo().search(
[('id', 'in', vals['taxes_id'][0][2])]):
total_tax += self.calculate_total_tax_percentage(tax)
data['tax'] = total_tax
if 'list_price' in vals.keys():
data['prices'] = [{'price': vals['list_price'],
'currency': self.env.company.currency_id.name}
]
pipedrive_product = self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'product'), ('odoo_ref', '=', self.id)])
if pipedrive_product and data:
headers = {
'Content-Type': 'application/json',
'Accept': 'application/json'
}
response = requests.put(
url=f'https://api.pipedrive.com/v1/products/'
f'{pipedrive_product.pipedrive_reference}',
params={
'api_token': self.env.user.company_id.api_key,
}, timeout=10, headers=headers, data=json.dumps(data))
if 'error' in response.json().keys():
raise ValidationError(
response.json()['error'])
return super().write(vals)
def calculate_total_tax_percentage(self, tax):
"""Method for calculating total tax"""
total_percentage_tax = 0.0
# Percentage Taxes
if tax.amount_type == 'percent':
total_percentage_tax = tax.amount
# Group Taxes
elif tax.amount_type == 'group':
for child_tax in tax.children_tax_ids.filtered(
lambda t: t.amount_type == 'percent'):
total_percentage_tax += child_tax.amount
# Fixed Taxes
elif tax.amount_type == 'fixed':
total_percentage_tax = (tax.amount / tax.list_price) * 100
# Division Taxes
else:
total_percentage_tax = (tax.list_price / tax.factor) * 100
return total_percentage_tax
def unlink(self):
"""Method for deleting a product from Pipedrive"""
pipedrive_product = self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'product'), ('odoo_ref', '=', self.id)])
if pipedrive_product:
headers = {
'Content-Type': 'application/json',
'Accept': 'application/json'
}
response = requests.delete(
url=f'https://api.pipedrive.com/v1/products/'
f'{pipedrive_product.pipedrive_reference}',
params={
'api_token': self.env.user.company_id.api_key,
}, timeout=10, headers=headers)
if 'error' in response.json().keys():
raise ValidationError(
response.json()['error'])
return super().unlink()
@api.model_create_multi
def create(self, vals_list):
"""Inherited to add the product to pipedrive"""
res = super().create(vals_list)
if self.env.company.product_synced and not res.pipedrive_reference:
if not self.env['pipedrive.record'].search(
[('record_type', '=', 'product',), ('odoo_ref', '=', res.id)]):
for product in vals_list:
tax_ids = product.get('taxes_id', [])
total_percentage_tax = 0.0
if tax_ids:
for tax_id in tax_ids[0][2]:
tax = self.env['account.tax'].sudo().browse(tax_id)
if tax.amount_type == 'percent':
total_percentage_tax += tax.amount
elif tax.amount_type == 'group':
for child_tax in tax.children_tax_ids:
if child_tax.amount_type == 'percent':
total_percentage_tax += child_tax.amount
elif tax.amount_type == 'fixed':
total_percentage_tax += (tax.amount / product[
'list_price']) * 100
data = {
'name': product['name'],
'tax': total_percentage_tax,
'code': product['default_code'],
'prices': [{
'price': product['list_price'],
'currency': self.env.company.currency_id.name,
'cost': product['standard_price']
}]
}
if 'uom_id' in product.keys():
data['unit'] = self.env['uom.uom'].sudo().browse(
product['uom_id']).name
response = requests.post(
url='https://api.pipedrive.com/v1/products',
params={
'api_token': self.env.company.api_key,
}, json=data, timeout=10)
if not response.json()['success']:
raise ValidationError(
response.json()['error'] + '. ' + response.json()[
'error_info'])
self.env['pipedrive.record'].sudo().create({
'pipedrive_reference': response.json()['data']['id'],
'record_type': 'product',
'odoo_ref': res.id
})
return res

434
odoo_pipedrive_connector/models/res_company.py

@ -0,0 +1,434 @@
# -*- coding: utf-8 -*-
################################################################################
#
# Cybrosys Technologies Pvt. Ltd.
#
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>).
# Author: Unnimaya C O (odoo@cybrosys.com)
#
# You can modify it under the terms of the GNU AFFERO
# GENERAL PUBLIC LICENSE (AGPL v3), Version 3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details.
#
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
# (AGPL v3) along with this program.
# If not, see <http://www.gnu.org/licenses/>.
#
################################################################################
import json
import requests
from odoo import fields, models, _
from odoo.exceptions import ValidationError
class ResCompany(models.Model):
"""Inherits Res Company for including Pipedrive credential fields"""
_inherit = 'res.company'
api_key = fields.Char(string='Token',
help="It is used to connect with Pipedrive"
)
product_synced = fields.Boolean(string='Product Synced', readonly=True,
help='True if the products are synced between Odoo and Pipedrive')
contact_synced = fields.Boolean(string='Contact Synced', readonly=True,
help='True if the contacts are synced between Odoo and Pipedrive')
lead_synced = fields.Boolean(string='Lead Synced', readonly=True,
help='True if the leads are synced between Odoo and Pipedrive')
is_delete_webhook = fields.Boolean(string='Is Delete Webhook',
help='To know whether the already '
'existing webhook is deleted or '
'not', default=False)
def delete_existing_webhook(self):
params = {
'api_token': self.api_key
}
existing_webhooks_response = requests.get(
"https://api.pipedrive.com/v1/webhooks", params=params)
if existing_webhooks_response.status_code == 200 and self.is_delete_webhook!=True:
existing_webhooks = existing_webhooks_response.json().get('data',
[])
for webhook in existing_webhooks:
webhook_id = webhook.get('id')
if webhook_id:
delete_response = requests.delete(
f"https://api.pipedrive.com/v1/webhooks/{webhook_id}",
params=params, timeout=10)
self.is_delete_webhook = True
return
def calculate_total_tax_percentage(self, product):
"""Method for calculating total tax"""
total_percentage_tax = 0.0
# Percentage Taxes
for tax in product.taxes_id.filtered(
lambda t: t.amount_type == 'percent'):
total_percentage_tax += tax.amount
# Group Taxes
for tax in product.taxes_id.filtered(
lambda t: t.amount_type == 'group'):
for child_tax in tax.children_tax_ids.filtered(
lambda t: t.amount_type == 'percent'):
total_percentage_tax += child_tax.amount
# Fixed Taxes
for tax in product.taxes_id.filtered(
lambda t: t.amount_type == 'fixed'):
total_percentage_tax += (tax.amount / product.list_price) * 100
# Division Taxes
for tax in product.taxes_id.filtered(
lambda t: t.amount_type == 'division'):
total_percentage_tax += (product.list_price / tax.factor) * 100
return total_percentage_tax
def action_sync_products(self):
"""Button action to sync products"""
if not self.api_key:
raise ValidationError(_('Please Enter an API Key'))
self.get_products()
self.export_products_to_pipedrive()
self.create_webhook(
'deleted',
'/delete_pipedrive_product',
'product')
self.create_webhook(
"updated",
'/update_pipedrive_product',
'product')
self.create_webhook(
"added",
'/add_pipedrive_product',
'product')
self.product_synced = True
def action_sync_contacts(self):
"""Button action to sync contacts"""
if not self.api_key:
raise ValidationError(_('Please Enter an API Key'))
self.get_contacts()
self.export_contacts_to_pipedrive()
self.create_webhook(
'updated', '/update_pipedrive_contact',
'person')
self.create_webhook(
'deleted', '/delete_pipedrive_contact',
'person')
self.create_webhook(
'added', '/add_pipedrive_contact',
'person')
self.contact_synced = True
def action_sync_leads(self):
"""Button action to sync leads"""
if not self.api_key:
raise ValidationError(_('Please Enter an API Key'))
self.get_leads()
self.export_leads_to_pipedrive()
self.lead_synced = True
def get_products(self):
"""Receives Products from Pipedrive"""
response = requests.get(url='https://api.pipedrive.com/v1/products',
params={
'api_token': self.api_key,
}, timeout=10)
if not response.json()['success']:
raise ValidationError(
response.json()['error'] + '. ' + response.json()[
'error_info'])
if response.json()['data']:
self.create_product_category()
for data in response.json()['data']:
if not self.env['pipedrive.record'].sudo().search(
[('pipedrive_reference', '=', str(data['id'])), ('record_type', '=', 'product')]):
uom_id = 1
if data['unit']:
for rec in self.env['uom.uom'].sudo().search([]).mapped(
'name'):
if rec.lower() == data['unit'].lower():
uom_id = self.env['uom.uom'].sudo().search(
[('name', '=', rec)]).id
if data['prices'][0]['price']:
currency = self.env['res.currency'].sudo().search(
[('name', '=', data['prices'][0]['currency']),
('active', 'in', [True, False])])
if not currency.active:
currency.active = True
product = self.env['product.template'].sudo().create({
'name': data['name'],
'description': data['description'],
'uom_id': uom_id,
'uom_po_id': uom_id,
'list_price': data['prices'][0]['price'],
'standard_price': data['prices'][0]['cost'],
'taxes_id': False,
'categ_id': self.env['pipedrive.record'].sudo().search(
[(
'pipedrive_reference', '=',
data['category'])])[0].odoo_ref
if self.env['pipedrive.record'].sudo().search(
[(
'pipedrive_reference', '=',
data['category'])]) else 1
})
if product:
self.env['pipedrive.record'].sudo().create({
'pipedrive_reference': data['id'],
'record_type': 'product',
'odoo_ref': product.id
})
product.taxes_id.unlink()
if data['tax'] != 0:
tax = self.env['account.tax'].sudo().search(
[('amount_type', '=', 'percent'),
('type_tax_use', '=', 'sale'), ('amount',
'=',
data['tax'])], limit=1)
if not tax:
tax = self.env['account.tax'].sudo().create({
'name': 'Tax ' + str(data['tax']) + '%',
'amount_type': 'percent',
'type_tax_use': 'sale',
'amount': data['tax']
})
product.sudo().write({
"taxes_id": [(4, tax.id)]
})
def create_product_category(self):
"""Returns product category from category_id"""
response = requests.get(
url='https://api.pipedrive.com/v1/productFields',
params={
'api_token': self.api_key,
}, timeout=10)
if not response.json()['success']:
raise ValidationError(
response.json()['error'] + '. ' + response.json()[
'error_info'])
for rec in response.json()['data']:
if rec['key'] == 'category':
for item in rec['options']:
category = self.env['product.category'].sudo().search(
[('name', '=', item['label'])])
if not category:
category = self.env['product.category'].sudo().create(
{
'name': item['label']
}
)
self.env['pipedrive.record'].sudo().create({
'pipedrive_reference': item['id'],
'record_type': 'categ',
'odoo_ref': category[0].id
})
elif not self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'categ'), ('odoo_ref', '=', category[0].id)]):
self.env['pipedrive.record'].sudo().create({
'pipedrive_reference': item['id'],
'record_type': 'categ',
'odoo_ref': category[0].id
})
def get_contacts(self):
"""Receives contacts from Pipedrive"""
response = requests.get(url='https://api.pipedrive.com/v1/persons',
params={
'api_token': self.api_key,
}, timeout=10)
if not response.json()['success']:
raise ValidationError(
response.json()['error'] + '. ' + response.json()[
'error_info'])
if response.json()['data']:
for data in response.json()['data']:
if not self.env['pipedrive.record'].sudo().search(
[('pipedrive_reference', '=', str(data['id'])), ('record_type', '=', 'partner')]):
partner = self.env['res.partner'].sudo().create({
'name': data['name'],
'phone': data['phone'][0]['value'],
'email': data['email'][0]['value']
})
if partner:
self.env['pipedrive.record'].sudo().create({
'pipedrive_reference': data['id'],
'record_type': 'partner',
'odoo_ref': partner.id
})
def get_leads(self):
"""Receives leads from Pipedrive"""
response = requests.get(url='https://api.pipedrive.com/v1/leads',
params={
'api_token': self.api_key,
}, timeout=10)
if not response.json()['success']:
raise ValidationError(
response.json()['error'] + '. ' + response.json()[
'error_info'])
if response.json()['data']:
for data in response.json()['data']:
expected_revenue = 0
if data['value']:
currency = self.env['res.currency'].sudo().search(
[('name', '=', data['value']['currency']),
('active', 'in', [True, False])])
if not currency.active:
currency.active = True
expected_revenue = currency.compute(
data['value']['amount'], self.env.company.currency_id)
if not self.env['pipedrive.record'].sudo().search(
[('pipedrive_reference', '=', str(data['id'])), ('record_type', '=', 'lead')]):
lead = self.env['crm.lead'].sudo().create({
'name': data['title'],
'type': 'opportunity',
'expected_revenue': expected_revenue
})
self.env['pipedrive.record'].sudo().create({
'pipedrive_reference': data['id'],
'record_type': 'lead',
'odoo_ref': lead.id
})
def export_products_to_pipedrive(self):
"""Export Products from Odoo to Pipedrive"""
pipedrive_products = self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'product')]).mapped('odoo_ref')
for product in self.env['product.template'].sudo().search(
[('id', 'not in', pipedrive_products)]):
data = {
'name': product.name,
'unit': product.uom_id.name,
'tax': self.calculate_total_tax_percentage(product),
'prices': [{
'price': product.list_price,
'currency': self.env.company.currency_id.name
}]
}
response = requests.post(
url='https://api.pipedrive.com/v1/products',
params={
'api_token': self.api_key,
}, json=data, timeout=10)
if not response.json()['success']:
raise ValidationError(
response.json()['error'] + '. ' + response.json()[
'error_info'])
self.env['pipedrive.record'].sudo().create({
'pipedrive_reference': response.json()['data']['id'],
'record_type': 'product',
'odoo_ref': product.id
})
def export_contacts_to_pipedrive(self):
"""Export Contacts from Odoo to Pipedrive"""
pipedrive_partner = self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'partner')]).mapped('odoo_ref')
for partner in self.env['res.partner'].sudo().search(
[('id', 'not in', pipedrive_partner)]):
self.create_contact(partner)
def create_contact(self, partner):
"""Create Persons in Pipedrive"""
data = {
'name': partner.name,
'email': partner.email,
'phone': partner.phone
}
response = requests.post(
url='https://api.pipedrive.com/v1/persons',
params={
'api_token': self.api_key,
}, json=data, timeout=10)
if 'success' in response.json(
).keys() and not response.json()['success'] and 'error' in \
response.json(
).keys():
raise ValidationError(
response.json()['error'])
self.env['pipedrive.record'].sudo().create({
'pipedrive_reference': response.json()['data']['id'],
'record_type': 'partner',
'odoo_ref': partner.id
})
def create_webhook(self, event_action, url, event_object):
"""Method for creating contact webhook in Pipedrive"""
self.delete_existing_webhook()
https_url = self.get_base_url().replace("http://", "https://")
payload = json.dumps({
"subscription_url": https_url + url,
"event_action": event_action,
"event_object": event_object
})
headers = {
'Content-Type': 'application/json',
'Accept': 'application/json'
}
params = {
'api_token': self.api_key,
}
existing_webhooks_response = requests.get("https://api.pipedrive.com/v1/webhooks", params=params)
webhook_found = False
if existing_webhooks_response.status_code == 200:
existing_webhooks = existing_webhooks_response.json().get('data', [])
for webhook in existing_webhooks:
if (webhook.get('event_action') == event_action and
webhook.get('event_object') == event_object and
webhook.get(
'subscription_url') == self.env['ir.config_parameter'].get_param(
'web.base.url') + url):
webhook_found = True
if not webhook_found:
response = requests.request("POST",
"https://api.pipedrive.com/v1/"
"webhooks",
headers=headers, data=payload,
params=params,
timeout=10)
if not response.json()['success']:
if 'error' in response.json().keys():
raise ValidationError(
response.json()['error'])
if 'message' in response.json().keys():
raise ValidationError(
response.json()['message'])
def export_leads_to_pipedrive(self):
"""Export Leads from Odoo to Pipedrive"""
pipedrive_lead = self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'lead')]).mapped('odoo_ref')
for lead in self.env['crm.lead'].sudo().search(
[('id', 'not in', pipedrive_lead)]):
if lead.partner_id:
if not self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'partner'), ('odoo_ref', '=', lead.partner_id.id)]):
self.create_contact(lead.partner_id)
data = {
'title': lead.name,
'person_id': int(self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'partner'), ('odoo_ref', '=', lead.partner_id.id)]).pipedrive_reference),
'value': {
'amount': lead.expected_revenue,
'currency': self.env.company.currency_id.name
}
}
response = requests.post(
url='https://api.pipedrive.com/v1/leads',
params={
'api_token': self.api_key,
}, json=data, timeout=10)
if not response.json()['success']:
raise ValidationError(
response.json()['error'] + '. ' + response.json()[
'error_info'])
self.env['pipedrive.record'].sudo().create({
'pipedrive_reference': response.json()['data']['id'],
'record_type': 'lead',
'odoo_ref': lead.id
})

113
odoo_pipedrive_connector/models/res_partner.py

@ -0,0 +1,113 @@
# -*- coding: utf-8 -*-
################################################################################
#
# Cybrosys Technologies Pvt. Ltd.
#
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>).
# Author: Unnimaya C O (odoo@cybrosys.com)
#
# You can modify it under the terms of the GNU AFFERO
# GENERAL PUBLIC LICENSE (AGPL v3), Version 3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details.
#
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
# (AGPL v3) along with this program.
# If not, see <http://www.gnu.org/licenses/>.
#
################################################################################
import json
import requests
from odoo import api, fields, models
from odoo.exceptions import ValidationError
class ResPartner(models.Model):
"""Inherits Res Users for including Pipedrive fields and fuctions"""
_inherit = 'res.partner'
pipedrive_reference = fields.Char(string='Pipedrive Id',
help="Pipedrive Id of the Partner")
def write(self, vals):
"""Inherited to update the partner details in Pipedrive"""
data = {}
if 'name' in vals.keys():
data['name'] = vals['name']
if 'email' in vals.keys():
data['email'] = [{'value': vals['email'], 'primary': True}]
if 'phone' in vals.keys():
data['phone'] = [{'value': vals['phone'], 'primary': True}]
pipedrive_contact = self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'partner'), ('odoo_ref', '=', self.id)])
if pipedrive_contact and data:
headers = {
'Content-Type': 'application/json',
'Accept': 'application/json'
}
response = requests.put(
url=f'https://api.pipedrive.com/v1/persons/'
f'{pipedrive_contact.pipedrive_reference}',
params={
'api_token': self.env.user.company_id.api_key,
}, timeout=10, headers=headers, data=json.dumps(data))
if 'error' in response.json().keys():
raise ValidationError(
response.json()['error'])
return super().write(vals)
def unlink(self):
"""Inherited to delete the partner from Pipedrive"""
headers = {
'Content-Type': 'application/json',
'Accept': 'application/json'
}
pipedrive_contact = self.env['pipedrive.record'].sudo().search(
[('record_type', '=', 'partner'), ('odoo_ref', '=', self.id)])
if pipedrive_contact:
response = requests.delete(
url=f'https://api.pipedrive.com/v1/persons/'
f'{pipedrive_contact.pipedrive_reference}',
params={
'api_token': self.env.user.company_id.api_key,
}, timeout=10, headers=headers)
if 'error' in response.json().keys():
raise ValidationError(
response.json()['error'])
return super().unlink()
@api.model_create_multi
def create(self, vals_list):
"""Inherited to add the contact to pipedrive"""
res = super().create(vals_list)
if self.env.company.contact_synced and not res.pipedrive_reference:
if not self.env['pipedrive.record'].search(
[('record_type', '=', 'partner',), ('odoo_ref', '=', res.id)]):
for partner in vals_list:
data = {
'name': partner['name']
}
if 'email' in partner.keys():
data['email'] = partner['email']
if 'phone' in partner.keys():
data['phone'] = partner['phone'],
response = requests.post(
url='https://api.pipedrive.com/v1/persons',
params={
'api_token': self.env.company.api_key,
}, json=data, timeout=10)
if 'success' in response.json(
).keys() and not response.json()['success'] and 'error' in \
response.json(
).keys():
raise ValidationError(
response.json()['error'])
self.env['pipedrive.record'].sudo().create({
'pipedrive_reference': response.json()['data']['id'],
'record_type': 'partner',
'odoo_ref': res.id
})
return res

2
odoo_pipedrive_connector/security/ir.model.access.csv

@ -0,0 +1,2 @@
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
access_pipedrive_record_user,access.pipedrive.record.user,model_pipedrive_record,base.group_user,1,1,1,1
1 id name model_id:id group_id:id perm_read perm_write perm_create perm_unlink
2 access_pipedrive_record_user access.pipedrive.record.user model_pipedrive_record base.group_user 1 1 1 1

BIN
odoo_pipedrive_connector/static/description/assets/icons/check.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.6 KiB

BIN
odoo_pipedrive_connector/static/description/assets/icons/chevron.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 310 B

BIN
odoo_pipedrive_connector/static/description/assets/icons/cogs.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

BIN
odoo_pipedrive_connector/static/description/assets/icons/consultation.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.4 KiB

BIN
odoo_pipedrive_connector/static/description/assets/icons/ecom-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 576 B

BIN
odoo_pipedrive_connector/static/description/assets/icons/education-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 733 B

BIN
odoo_pipedrive_connector/static/description/assets/icons/hotel-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 911 B

BIN
odoo_pipedrive_connector/static/description/assets/icons/license.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

BIN
odoo_pipedrive_connector/static/description/assets/icons/lifebuoy.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.2 KiB

BIN
odoo_pipedrive_connector/static/description/assets/icons/logo.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.4 KiB

BIN
odoo_pipedrive_connector/static/description/assets/icons/manufacturing-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 673 B

BIN
odoo_pipedrive_connector/static/description/assets/icons/pos-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 878 B

BIN
odoo_pipedrive_connector/static/description/assets/icons/puzzle.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 653 B

BIN
odoo_pipedrive_connector/static/description/assets/icons/restaurant-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 905 B

BIN
odoo_pipedrive_connector/static/description/assets/icons/service-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 839 B

BIN
odoo_pipedrive_connector/static/description/assets/icons/trading-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 427 B

BIN
odoo_pipedrive_connector/static/description/assets/icons/training.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 627 B

BIN
odoo_pipedrive_connector/static/description/assets/icons/update.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.2 KiB

BIN
odoo_pipedrive_connector/static/description/assets/icons/user.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 988 B

BIN
odoo_pipedrive_connector/static/description/assets/icons/wrench.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.2 KiB

BIN
odoo_pipedrive_connector/static/description/assets/misc/categories.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.5 KiB

BIN
odoo_pipedrive_connector/static/description/assets/misc/check-box.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

BIN
odoo_pipedrive_connector/static/description/assets/misc/compass.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.9 KiB

BIN
odoo_pipedrive_connector/static/description/assets/misc/corporate.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

BIN
odoo_pipedrive_connector/static/description/assets/misc/customer-support.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.1 KiB

BIN
odoo_pipedrive_connector/static/description/assets/misc/cybrosys-logo.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.4 KiB

BIN
odoo_pipedrive_connector/static/description/assets/misc/features.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 589 B

BIN
odoo_pipedrive_connector/static/description/assets/misc/logo.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.4 KiB

BIN
odoo_pipedrive_connector/static/description/assets/misc/pictures.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.7 KiB

BIN
odoo_pipedrive_connector/static/description/assets/misc/pie-chart.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.3 KiB

BIN
odoo_pipedrive_connector/static/description/assets/misc/right-arrow.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 967 B

BIN
odoo_pipedrive_connector/static/description/assets/misc/star.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.6 KiB

BIN
odoo_pipedrive_connector/static/description/assets/misc/support.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.8 KiB

BIN
odoo_pipedrive_connector/static/description/assets/misc/whatsapp.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.0 KiB

BIN
odoo_pipedrive_connector/static/description/assets/modules/agriculture.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 68 KiB

BIN
odoo_pipedrive_connector/static/description/assets/modules/hospital.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 65 KiB

BIN
odoo_pipedrive_connector/static/description/assets/modules/l1.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 144 KiB

BIN
odoo_pipedrive_connector/static/description/assets/modules/l2.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 257 KiB

BIN
odoo_pipedrive_connector/static/description/assets/modules/l3.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 88 KiB

BIN
odoo_pipedrive_connector/static/description/assets/modules/l4.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 147 KiB

BIN
odoo_pipedrive_connector/static/description/assets/modules/l5.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 146 KiB

BIN
odoo_pipedrive_connector/static/description/assets/modules/l6.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 74 KiB

BIN
odoo_pipedrive_connector/static/description/assets/modules/project_dashboard.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 65 KiB

BIN
odoo_pipedrive_connector/static/description/assets/modules/shopify.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 70 KiB

BIN
odoo_pipedrive_connector/static/description/assets/modules/systray.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 96 KiB

BIN
odoo_pipedrive_connector/static/description/assets/modules/woocomp_connector.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 60 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/1.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 107 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/2.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 207 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 89 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot0.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 140 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot1.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 80 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot10.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 58 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot11.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 92 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot12.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 53 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot13.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 182 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot14.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 53 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot15.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 230 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot16.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 103 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot17.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 65 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot18.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 68 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot19.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 103 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot2.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 72 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot20.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 76 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot21.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 105 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot22.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 125 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot23.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 81 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot24.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 89 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot25.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 191 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot3.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 108 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot4.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 86 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot6.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 88 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot7.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 98 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot8.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 56 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/Screenshot9.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 68 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/hero.gif

Binary file not shown.

After

Width:  |  Height:  |  Size: 440 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/sync_contact.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 70 KiB

BIN
odoo_pipedrive_connector/static/description/assets/screenshots/sync_lead.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 67 KiB

BIN
odoo_pipedrive_connector/static/description/banner.jpg

Binary file not shown.

After

Width:  |  Height:  |  Size: 96 KiB

BIN
odoo_pipedrive_connector/static/description/icon.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 16 KiB

795
odoo_pipedrive_connector/static/description/index.html

@ -0,0 +1,795 @@
<div style="background-color: #714B67; min-height: 600px; width: 100%; padding: 15px; position: relative;">
<!-- TITLE BAR -->
<div class="d-flex align-items-center justify-content-between"
style="border-bottom: 1px solid #875A7B; padding: 15px; display: flex; justify-content: space-between; align-items: center;">
<img src="assets/misc/cybrosys-logo.png" width="42" height="42"
style="width: 42px; height: 42px;"/>
<div>
<div style="color: #7C7BAD; font-size: 14px; font-family: 'Montserrat', sans-serif; font-weight: bold; background-color: white; display: inline-block; padding: 3px 10px; border-radius: 50px;"
class="mr-2">
<i class="fa fa-check mr-1"></i>Community
</div>
<div style="color: #7C7BAD; font-size: 14px; font-family: 'Montserrat', sans-serif; font-weight: bold; background-color: white; display: inline-block; padding: 3px 10px; border-radius: 50px;"
class="mr-2">
<i class="fa fa-check mr-1"></i>Enterprise
</div>
<div style="color: #017E84; font-size: 14px; font-family: 'Montserrat', sans-serif; font-weight: bold; background-color: white; display: inline-block; padding: 3px 10px; border-radius: 50px;"
class="mr-2">
<i class="fa fa-check mr-1"></i>Odoo.sh
</div>
</div>
</div>
<!-- END OF TITLE BAR -->
<div class="container">
<div class="row">
<div class="col-sm-12 col-md-12 col-lg-12">
<!-- APP HERO -->
<h1 style="color: #FFFFFF; font-weight: bolder; font-size: 50px; text-align: center; margin-top: 50px;">
Odoo Pipedrive Connector</h1>
<p style="color:#FFFFFF; padding: 8px 15px; text-align: center; font-size: 24px;">
Integrate Contacts, Products and Leads between Pipedrive and
Odoo.</p>
<!-- END OF APP HERO -->
<img src="assets/screenshots/hero.gif"
style="width: 75%; height: auto; position: absolute; margin-left: auto; margin-right: auto; top: 45%; left: 12%; right: auto;"/>
</div>
</div>
</div>
</div>
<!-- NAVIGATION SECTION -->
<div class="d-flex align-items-center"
style="border-bottom: 2px solid #714B67; padding: 15px 0px; margin-top: 300px;">
<div class="d-flex justify-content-center align-items-center mr-2"
style="background-color: #F5F5F5; border-radius: 0px; width: 40px; height: 40px;">
<img src="assets/misc/compass.png"/>
</div>
<h2 class="mt-2"
style="font-family: 'Montserrat', sans-serif; font-size: 24px; font-weight: bold;">
Explore This
Module</h2>
</div>
<div class="row my-4" style="font-family: 'Montserrat', sans-serif;">
<div class="col-sm-12 col-md-6 my-3">
<a href="#overview">
<div class="d-flex justify-content-between align-items-center"
style="background-color: #f5f5f5; padding: 30px; width: 100%;">
<div>
<span style="color: #714B67; font-size: 24px; font-weight: 500; display: block;">Overview</span>
<span
style="color: #714B67; font-size: 16px; font-weight: 400; color:#282F33; display: block;">Learn
more about this
module</span>
</div>
<img src="assets/misc/right-arrow.png" width="36" height="36"/>
</div>
</a>
</div>
<div class="col-sm-12 col-md-6 my-3">
<a href="#features">
<div class="d-flex justify-content-between align-items-center"
style="background-color: #f5f5f5; padding: 30px; width: 100%;">
<div>
<span style="color: #714B67; font-size: 24px; font-weight: 500; display: block;">Features</span>
<span
style="color: #714B67; font-size: 16px; font-weight: 400; color:#282F33; display: block;">View
features of this
module</span>
</div>
<img src="assets/misc/right-arrow.png" width="36" height="36"/>
</div>
</a>
</div>
<div class="col-sm-12 col-md-6 my-3">
<a href="#screenshots">
<div class="d-flex justify-content-between align-items-center"
style="background-color: #f5f5f5; padding: 30px; width: 100%;">
<div>
<span style="color: #714B67; font-size: 24px; font-weight: 500; display: block;">Screenshots</span>
<span
style="color: #714B67; font-size: 16px; font-weight: 400; color:#282F33; display: block;">View
screenshots for this
module</span>
</div>
<img src="assets/misc/right-arrow.png" width="36" height="36"/>
</div>
</a>
</div>
</div>
<!-- END OF NAVIGATION SECTION -->
<!-- OVERVIEW SECTION -->
<div class="d-flex align-items-center"
style="border-bottom: 2px solid #714B67; padding: 15px 0px;" id="overview">
<div class="d-flex justify-content-center align-items-center mr-2"
style="background-color: #F5F5F5; border-radius: 0px; width: 40px; height: 40px;">
<img src="assets/misc/pie-chart.png"/>
</div>
<h2 class="mt-2"
style="font-family: 'Montserrat', sans-serif; font-size: 24px; font-weight: bold;">
Overview
</h2>
</div>
<div class="row"
style="font-family: 'Montserrat', sans-serif; font-weight: 400; font-size: 14px; line-height: 200%;">
<div class="col-sm-12 py-4">
<p style="font-family: 'Roboto', sans-serif !important; font-weight: 400 !important; color: #282F33 !important; font-size: 1rem !important; line-height: 30px !important;">
This module helps to successfully import all products,
contacts and leads between Pipedrive to Odoo. Also, it is
possible to export all these data from Odoo to Pipedrive.
All import and export operations can be performed in a single
button click.
<mark>
Please note that this feature is only supported on secure (HTTPS) URLs.
</mark>
</p>
</div>
</div>
<!-- END OF OVERVIEW SECTION -->
<!-- FEATURES SECTION -->
<div class="d-flex align-items-center"
style="border-bottom: 2px solid #714B67; padding: 15px 0px;" id="features">
<div class="d-flex justify-content-center align-items-center mr-2"
style="background-color: #F5F5F5; border-radius: 0px; width: 40px; height: 40px;">
<img src="assets/misc/features.png"/>
</div>
<h2 class="mt-2"
style="font-family: 'Montserrat', sans-serif; font-size: 24px; font-weight: bold;">
Features
</h2>
</div>
<div class="row"
style="font-family: 'Montserrat', sans-serif; font-weight: 400; font-size: 14px; line-height: 200%;">
<div class="col-sm-12 col-md-6">
<div class="d-flex align-items-center"
style="margin-top: 40px; margin-bottom: 40px">
<img src="assets/misc/check-box.png" class="mr-2"/>
<span
style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Exports all Products, Contacts and Leads from Odoo to
Pipedrive.
</span>
</div>
<div class="d-flex align-items-center"
style="margin-top: 40px; margin-bottom: 40px">
<img src="assets/misc/check-box.png" class="mr-2"/>
<span
style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Imports all Products, Contacts and Leads from Pipedrive to
Odoo.
</span>
</div>
<div class="d-flex align-items-center"
style="margin-top: 30px; margin-bottom: 30px">
<img src="assets/misc/check-box.png" class="mr-2"/>
<span
style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Sync Product updates and deletions between Pipedrive and
Odoo. </span>
</div>
<div class="d-flex align-items-center"
style="margin-top: 30px; margin-bottom: 30px">
<img src="assets/misc/check-box.png" class="mr-2"/>
<span
style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Sync Contact updates and deletions between Pipedrive and
Odoo. </span>
</div> <div class="d-flex align-items-center"
style="margin-top: 30px; margin-bottom: 30px">
<img src="assets/misc/check-box.png" class="mr-2"/>
<span
style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Sync Lead updates and deletions from Odoo to Pipedrive. </span>
</div>
</div>
</div>
<!-- END OF FEATURES SECTION -->
<!-- SCREENSHOTS SECTION -->
<div class="d-flex align-items-center"
style="border-bottom: 2px solid #714B67; padding: 15px 0px;">
<div class="d-flex justify-content-center align-items-center mr-2"
style="background-color: #F5F5F5; border-radius: 0px; width: 40px; height: 40px;">
<img src="assets/misc/pictures.png"/>
</div>
<h2 class="mt-2"
style="font-family: 'Montserrat', sans-serif; font-size: 24px; font-weight: bold;">
Screenshots
</h2>
</div>
<!--SCREENSHOTS-->
<div class="row" id="screenshots">
<div class="col-sm-12">
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Login to <a
href="https://demo-sandbox4.pipedrive.com/settings/api">Pipedrive
Sandbox Account
</a></h3>
<img src="assets/screenshots/Screenshot.png" class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Create Token</h3>
<p style="font-weight: 400; font-family: 'Montserrat', sans-serif; font-size: 14px;">
Click on your profile picture at the right corner. Select the
option Personal preferences. Under API tab, you can see
Your personal API token.</p>
<img src="assets/screenshots/Screenshot0.png" class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Create new Product in Pipedrive</h3>
<img src="assets/screenshots/Screenshot1.png"
class="img-thumbnail"/>
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Sync Products</h3>
<p
style="font-weight: 400; font-family: 'Montserrat', sans-serif; font-size: 14px;">
Add Token and click SYNC PRODUCTS.
</p>
<img src="assets/screenshots/Screenshot2.png" class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Imported Product in Odoo</h3>
<img src="assets/screenshots/Screenshot3.png" class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Exported Products in Pipedrive</h3>
<img src="assets/screenshots/Screenshot11.png"
class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Create a Person in Pipedrive</h3>
<img src="assets/screenshots/Screenshot4.png" class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Sync Contacts</h3>
<p
style="font-weight: 400; font-family: 'Montserrat', sans-serif; font-size: 14px;">
Add Token and click SYNC CONTACTS.
</p>
<img src="assets/screenshots/sync_contact.png" class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Imported Contact in Odoo</h3>
<img src="assets/screenshots/Screenshot6.png" class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Exported Contacts in Pipedrive</h3>
<img src="assets/screenshots/Screenshot13.png"
class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Create a Lead in Pipedrive</h3>
<img src="assets/screenshots/Screenshot7.png" class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Sync Leads</h3>
<p
style="font-weight: 400; font-family: 'Montserrat', sans-serif; font-size: 14px;">
Add Token and click SYNC LEADS.
</p>
<img src="assets/screenshots/sync_lead.png" class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Imported Lead in Odoo</h3>
<img src="assets/screenshots/Screenshot9.png" class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Exported Leads in Pipedrive</h3>
<img src="assets/screenshots/Screenshot15.png"
class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Update Product Name from Odoo</h3>
<img src="assets/screenshots/Screenshot16.png"
class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Product Name updated in Pipedrive</h3>
<img src="assets/screenshots/Screenshot17.png"
class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Product UNIT PRICE updated from Pipedrive</h3>
<p
style="font-weight: 400; font-family: 'Montserrat', sans-serif; font-size: 14px;">
Since Odoo utilizes various types of taxes that are not
represented as simple numerical values, while Pipedrive provides
tax information as numerical values, synchronizing tax updates
for products from Pipedrive to Odoo is currently not feasible
</p>
<img src="assets/screenshots/Screenshot18.png"
class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Product Sales Price updated in Odoo</h3>
<img src="assets/screenshots/Screenshot19.png"
class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Contact's Email updated from Odoo</h3>
<img src="assets/screenshots/Screenshot20.png"
class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Contact's Email updated in Pipedrive.</h3>
<img src="assets/screenshots/Screenshot21.png"
class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Contact's Name updated from Pipedrive</h3>
<img src="assets/screenshots/Screenshot22.png"
class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Contact's Name updated in Odoo.</h3>
<img src="assets/screenshots/Screenshot23.png"
class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Updated the Expected Revenue of Lead from Odoo</h3>
<p
style="font-weight: 400; font-family: 'Montserrat', sans-serif; font-size: 14px;">
Due to the lack of webhook support for leads in Pipedrive,
synchronizing lead updates from Pipedrive to Odoo is currently
not feasible
</p>
<img src="assets/screenshots/Screenshot24.png"
class="img-thumbnail">
</div>
<div style="display: block; margin: 30px auto;">
<h3 style="font-family: 'Montserrat', sans-serif; font-size: 18px; font-weight: bold;">
Updated the Value of Lead updated from Odoo</h3>
<img src="assets/screenshots/Screenshot25.png"
class="img-thumbnail">
</div>
</div>
</div>
<!-- END OF SCREENSHOTS SECTION -->
<!-- RELATED PRODUCTS -->
<div class="d-flex align-items-center"
style="border-bottom: 2px solid #714B67; padding: 15px 0px;">
<div class="d-flex justify-content-center align-items-center mr-2"
style="background-color: #F5F5F5; border-radius: 0px; width: 40px; height: 40px;">
<img src="assets/misc/categories.png"/>
</div>
<h2 class="mt-2"
style="font-family: 'Montserrat', sans-serif; font-size: 24px; font-weight: bold;">
Related
Products
</h2>
</div>
<div class="row">
<div class="col-sm-12">
<div id="demo1" class="row carousel slide" data-ride="carousel">
<!-- The slideshow -->
<div class="carousel-inner" style="padding: 30px;">
<div class="carousel-item" style="min-height: 198.656px;">
<div class="col-xs-12 col-sm-4 col-md-4 mb16 mt16"
style="float:left">
<a href="https://apps.odoo.com/apps/modules/16.0/upcitemdb_integration/"
target="_blank">
<div style="border-radius:10px">
<img class="img img-responsive center-block"
style="border-radius: 0px;"
src="assets/modules/l1.png">
</div>
</a>
</div>
<div class="col-xs-12 col-sm-4 col-md-4 mb16 mt16"
style="float:left">
<a href="https://apps.odoo.com/apps/modules/16.0/education_erp_dashboard/"
target="_blank">
<div style="border-radius:10px">
<img class="img img-responsive center-block"
style="border-radius: 0px;"
src="assets/modules/l2.png">
</div>
</a>
</div>
<div class="col-xs-12 col-sm-4 col-md-4 mb16 mt16"
style="float:left">
<a href="https://apps.odoo.com/apps/modules/16.0/agriculture_management_odoo/"
target="_blank">
<div style="border-radius:10px">
<img class="img img-responsive center-block"
style="border-radius: 0px;"
src="assets/modules/l3.png">
</div>
</a>
</div>
</div>
<div class="carousel-item active"
style="min-height: 198.656px;">
<div class="col-xs-12 col-sm-4 col-md-4 mb16 mt16"
style="float:left">
<a href="https://apps.odoo.com/apps/modules/16.0/product_image_suggestion/"
target="_blank">
<div style="border-radius:10px">
<img class="img img-responsive center-block"
style="border-radius: 0px;"
src="assets/modules/l4.png">
</div>
</a>
</div>
<div class="col-xs-12 col-sm-4 col-md-4 mb16 mt16"
style="float:left">
<a href="https://apps.odoo.com/apps/modules/16.0/odoo_magento_2_4_5/"
target="_blank">
<div style="border-radius:10px">
<img class="img img-responsive center-block"
style="border-radius: 0px;"
src="assets/modules/l5.png">
</div>
</a>
</div>
<div class="col-xs-12 col-sm-4 col-md-4 mb16 mt16"
style="float:left">
<a href="https://apps.odoo.com/apps/modules/16.0/advanced_dynamic_dashboard/"
target="_blank">
<div style="border-radius:10px">
<img class="img img-responsive center-block"
style="border-radius: 0px;"
src="assets/modules/l6.png">
</div>
</a>
</div>
</div>
</div>
<!-- Left and right controls -->
<a class="carousel-control-prev" href="#demo1" data-slide="prev"
style="width:35px; color:#000"> <span
class="carousel-control-prev-icon"><i
class="fa fa-chevron-left"
style="font-size:24px"></i></span>
</a> <a class="carousel-control-next" href="#demo1"
data-slide="next" style="width:35px; color:#000">
<span class="carousel-control-next-icon"><i
class="fa fa-chevron-right"
style="font-size:24px"></i></span>
</a>
</div>
</div>
</div>
<!-- END OF RELATED PRODUCTS -->
<!-- OUR SERVICES -->
<div class="d-flex align-items-center"
style="border-bottom: 2px solid #714B67; padding: 15px 0px;">
<div class="d-flex justify-content-center align-items-center mr-2"
style="background-color: #F5F5F5; border-radius: 0px; width: 40px; height: 40px;">
<img src="assets/misc/star.png"/>
</div>
<h2 class="mt-2"
style="font-family: 'Montserrat', sans-serif; font-size: 24px; font-weight: bold;">
Our Services
</h2>
</div>
<div class="container my-5">
<div class="row">
<div class="col-lg-4 d-flex flex-column justify-content-center align-items-center my-4">
<div class="d-flex justify-content-center align-items-center mx-3 my-3"
style="background-color: #1dd1a1 !important; border-radius: 15px !important; height: 80px; width: 80px;">
<img src="assets/icons/cogs.png" class="img-responsive"
height="48px" width="48px">
</div>
<h6 class="text-center"
style="font-family: Montserrat, 'sans-serif' !important; font-weight: bold;">
Odoo
Customization</h6>
</div>
<div class="col-lg-4 d-flex flex-column justify-content-center align-items-center my-4">
<div class="d-flex justify-content-center align-items-center mx-3 my-3"
style="background-color: #ff6b6b !important; border-radius: 15px !important; height: 80px; width: 80px;">
<img src="assets/icons/wrench.png"
class="img-responsive"
height="48px" width="48px">
</div>
<h6 class="text-center"
style="font-family: Montserrat, 'sans-serif' !important; font-weight: bold;">
Odoo
Implementation</h6>
</div>
<div class="col-lg-4 d-flex flex-column justify-content-center align-items-center my-4">
<div class="d-flex justify-content-center align-items-center mx-3 my-3"
style="background-color: #6462CD !important; border-radius: 15px !important; height: 80px; width: 80px;">
<img src="assets/icons/lifebuoy.png"
class="img-responsive"
height="48px" width="48px">
</div>
<h6 class="text-center"
style="font-family: Montserrat, 'sans-serif' !important; font-weight: bold;">
Odoo
Support</h6>
</div>
<div class="col-lg-4 d-flex flex-column justify-content-center align-items-center my-4">
<div class="d-flex justify-content-center align-items-center mx-3 my-3"
style="background-color: #ffa801 !important; border-radius: 15px !important; height: 80px; width: 80px;">
<img src="assets/icons/user.png" class="img-responsive"
height="48px" width="48px">
</div>
<h6 class="text-center"
style="font-family: Montserrat, 'sans-serif' !important; font-weight: bold;">
Hire
Odoo
Developer</h6>
</div>
<div class="col-lg-4 d-flex flex-column justify-content-center align-items-center my-4">
<div class="d-flex justify-content-center align-items-center mx-3 my-3"
style="background-color: #54a0ff !important; border-radius: 15px !important; height: 80px; width: 80px;">
<img src="assets/icons/puzzle.png"
class="img-responsive"
height="48px" width="48px">
</div>
<h6 class="text-center"
style="font-family: Montserrat, 'sans-serif' !important; font-weight: bold;">
Odoo
Integration</h6>
</div>
<div class="col-lg-4 d-flex flex-column justify-content-center align-items-center my-4">
<div class="d-flex justify-content-center align-items-center mx-3 my-3"
style="background-color: #6d7680 !important; border-radius: 15px !important; height: 80px; width: 80px;">
<img src="assets/icons/update.png"
class="img-responsive"
height="48px" width="48px">
</div>
<h6 class="text-center"
style="font-family: Montserrat, 'sans-serif' !important; font-weight: bold;">
Odoo
Migration</h6>
</div>
<div class="col-lg-4 d-flex flex-column justify-content-center align-items-center my-4">
<div class="d-flex justify-content-center align-items-center mx-3 my-3"
style="background-color: #786fa6 !important; border-radius: 15px !important; height: 80px; width: 80px;">
<img src="assets/icons/consultation.png"
class="img-responsive"
height="48px" width="48px">
</div>
<h6 class="text-center"
style="font-family: Montserrat, 'sans-serif' !important; font-weight: bold;">
Odoo
Consultancy</h6>
</div>
<div class="col-lg-4 d-flex flex-column justify-content-center align-items-center my-4">
<div class="d-flex justify-content-center align-items-center mx-3 my-3"
style="background-color: #f8a5c2 !important; border-radius: 15px !important; height: 80px; width: 80px;">
<img src="assets/icons/training.png"
class="img-responsive"
height="48px" width="48px">
</div>
<h6 class="text-center"
style="font-family: Montserrat, 'sans-serif' !important; font-weight: bold;">
Odoo
Implementation</h6>
</div>
<div class="col-lg-4 d-flex flex-column justify-content-center align-items-center my-4">
<div class="d-flex justify-content-center align-items-center mx-3 my-3"
style="background-color: #e6be26 !important; border-radius: 15px !important; height: 80px; width: 80px;">
<img src="assets/icons/license.png"
class="img-responsive"
height="48px" width="48px">
</div>
<h6 class="text-center"
style="font-family: Montserrat, 'sans-serif' !important; font-weight: bold;">
Odoo
Licensing Consultancy</h6>
</div>
</div>
</div>
<!--END OF OUR SERVICES -->
<!-- OUR INDUSTRIES -->
<div class="d-flex align-items-center"
style="border-bottom: 2px solid #714B67; padding: 15px 0px;">
<div class="d-flex justify-content-center align-items-center mr-2"
style="background-color: #F5F5F5; border-radius: 0px; width: 40px; height: 40px;">
<img src="assets/misc/corporate.png"/>
</div>
<h2 class="mt-2"
style="font-family: 'Montserrat', sans-serif; font-size: 24px; font-weight: bold;">
Our
Industries
</h2>
</div>
<div class="container my-5">
<div class="row">
<div class="col-lg-3">
<div class="my-4 d-flex flex-column justify-content-center"
style="background-color: #f6f8f9 !important; border-radius: 0px; padding: 2rem !important; height: 250px !important;">
<img src="assets/icons/trading-black.png"
class="img-responsive mb-3" height="48px"
width="48px">
<h5 style="font-family: Montserrat, sans-serif !important; color: #000 !important; font-weight: bold;">
Trading
</h5>
<p style="font-family: Montserrat, sans-serif !important; font-size: 0.9rem !important;">
Easily procure
and
sell your products</p>
</div>
</div>
<div class="col-lg-3">
<div class="my-4 d-flex flex-column justify-content-center"
style="background-color: #f6f8f9 !important; border-radius: 0px; padding: 2rem !important; height: 250px !important;">
<img src="assets/icons/pos-black.png"
class="img-responsive mb-3" height="48px"
width="48px">
<h5 style="font-family: Montserrat, sans-serif !important; color: #000 !important; font-weight: bold;">
POS
</h5>
<p style="font-family: Montserrat, sans-serif !important; font-size: 0.9rem !important;">
Easy
configuration
and convivial experience</p>
</div>
</div>
<div class="col-lg-3">
<div class="my-4 d-flex flex-column justify-content-center"
style="background-color: #f6f8f9 !important; border-radius: 0px; padding: 2rem !important; height: 250px !important;">
<img src="assets/icons/education-black.png"
class="img-responsive mb-3" height="48px"
width="48px">
<h5 style="font-family: Montserrat, sans-serif !important; color: #000 !important; font-weight: bold;">
Education
</h5>
<p style="font-family: Montserrat, sans-serif !important; font-size: 0.9rem !important;">
A platform for
educational management</p>
</div>
</div>
<div class="col-lg-3">
<div class="my-4 d-flex flex-column justify-content-center"
style="background-color: #f6f8f9 !important; border-radius: 0px; padding: 2rem !important; height: 250px !important;">
<img src="assets/icons/manufacturing-black.png"
class="img-responsive mb-3" height="48px"
width="48px">
<h5 style="font-family: Montserrat, sans-serif !important; color: #000 !important; font-weight: bold;">
Manufacturing
</h5>
<p style="font-family: Montserrat, sans-serif !important; font-size: 0.9rem !important;">
Plan, track and
schedule your operations</p>
</div>
</div>
<div class="col-lg-3">
<div class="my-4 d-flex flex-column justify-content-center"
style="background-color: #f6f8f9 !important; border-radius: 0px; padding: 2rem !important; height: 250px !important;">
<img src="assets/icons/ecom-black.png"
class="img-responsive mb-3" height="48px"
width="48px">
<h5 style="font-family: Montserrat, sans-serif !important; color: #000 !important; font-weight: bold;">
E-commerce &amp; Website
</h5>
<p style="font-family: Montserrat, sans-serif !important; font-size: 0.9rem !important;">
Mobile
friendly,
awe-inspiring product pages</p>
</div>
</div>
<div class="col-lg-3">
<div class="my-4 d-flex flex-column justify-content-center"
style="background-color: #f6f8f9 !important; border-radius: 0px; padding: 2rem !important; height: 250px !important;">
<img src="assets/icons/service-black.png"
class="img-responsive mb-3" height="48px"
width="48px">
<h5 style="font-family: Montserrat, sans-serif !important; color: #000 !important; font-weight: bold;">
Service Management
</h5>
<p style="font-family: Montserrat, sans-serif !important; font-size: 0.9rem !important;">
Keep track of
services and invoice</p>
</div>
</div>
<div class="col-lg-3">
<div class="my-4 d-flex flex-column justify-content-center"
style="background-color: #f6f8f9 !important; border-radius: 0px; padding: 2rem !important; height: 250px !important;">
<img src="assets/icons/restaurant-black.png"
class="img-responsive mb-3" height="48px"
width="48px">
<h5 style="font-family: Montserrat, sans-serif !important; color: #000 !important; font-weight: bold;">
Restaurant
</h5>
<p style="font-family: Montserrat, sans-serif !important; font-size: 0.9rem !important;">
Run your bar or
restaurant methodically</p>
</div>
</div>
<div class="col-lg-3">
<div class="my-4 d-flex flex-column justify-content-center"
style="background-color: #f6f8f9 !important; border-radius: 0px; padding: 2rem !important; height: 250px !important;">
<img src="assets/icons/hotel-black.png"
class="img-responsive mb-3" height="48px"
width="48px">
<h5 style="font-family: Montserrat, sans-serif !important; color: #000 !important; font-weight: bold;">
Hotel Management
</h5>
<p style="font-family: Montserrat, sans-serif !important; font-size: 0.9rem !important;">
An
all-inclusive
hotel management application</p>
</div>
</div>
</div>
</div>
<!-- END OF OUR INDUSTRIES -->
<!-- SUPPORT -->
<div class="d-flex align-items-center"
style="border-bottom: 2px solid #714B67; padding: 15px 0px;">
<div class="d-flex justify-content-center align-items-center mr-2"
style="background-color: #F5F5F5; border-radius: 0px; width: 40px; height: 40px;">
<img src="assets/misc/customer-support.png"/>
</div>
<h2 class="mt-2"
style="font-family: 'Montserrat', sans-serif; font-size: 24px; font-weight: bold;">
Support
</h2>
</div>
<div class="container mt-5">
<div class="row">
<div class="col-sm-12 col-md-6">
<div style="background-color: #F6F8F9; padding: 30px; display: flex; align-items: center;">
<div class="mr-4"
style="background-color: #714B67; display: inline-block; height: 70px; width: 70px; display: flex; align-items: center; justify-content: center;">
<img src="assets/misc/support.png" height="48"
width="48"
style="width: 42px; height: 42px;"/>
</div>
<div>
<h4>Need Help?</h4>
<p style="line-height: 100%;">Got questions or need
help?
Get in touch.</p>
<a href="mailto:odoo@cybrosys.com">
<p style="font-weight: 400; font-size: 28px; line-height: 80%; color: #714B67;">
odoo@cybrosys.com</p>
</a>
</div>
</div>
</div>
<div class="col-sm-12 col-md-6">
<div style="background-color: #F6F8F9; padding: 30px; display: flex; align-items: center;">
<div class="mr-4"
style="background-color: #2AC44D; display: inline-block; height: 70px; width: 70px; display: flex; align-items: center; justify-content: center;">
<img src="assets/misc/whatsapp.png" height="52"
width="52"
style="width: 52px; height: 52px;"/>
</div>
<div>
<h4>WhatsApp</h4>
<p style="line-height: 100%;">Say hi to us on
WhatsApp!</p>
<a href="https://api.whatsapp.com/send?phone=918606827707">
<p style="font-weight: 400; font-size: 28px; line-height: 80%; color: #714B67;">
+91 86068
27707</p>
</a>
</div>
</div>
</div>
</div>
<div class="row">
<div class="col-sm-12 my-5 d-flex justify-content-center align-items-center">
<img src="assets/misc/logo.png" width="144" height="31"
style="width:144px; height: 31px; margin-top: 40px;"/>
</div>
</div>
</div>
<!-- END OF SUPPORT -->

47
odoo_pipedrive_connector/views/res_company_views.xml

@ -0,0 +1,47 @@
<?xml version="1.0" encoding="UTF-8" ?>
<odoo>
<!-- Inherit res.company to add pipedrive fields -->
<record id="view_company_form" model="ir.ui.view">
<field name="name">
res.company.view.form.inherit.odoo.pipedrive.connector
</field>
<field name="inherit_id" ref="base.view_company_form"/>
<field name="model">res.company</field>
<field name="arch" type="xml">
<xpath expr="//notebook" position="inside">
<page string="Pipedrive" name="Pipedrive">
<group>
<field name="api_key" password="True"
string="Token"/>
<field name="is_delete_webhook"/>
</group>
<group>
<group>
<field name="product_synced" widget="boolean_toggle"/>
<field name="contact_synced" widget="boolean_toggle"/>
<field name="lead_synced" widget="boolean_toggle"/>
</group>
</group>
<group col="6">
<group>
<button string="Sync Products" class="btn-primary"
attrs="{'invisible': [('product_synced', '=', True)]}"
type="object" name="action_sync_products" style="margin:10px"/>
</group>
<group>
<button string="Sync Contacts" class="btn-primary" style="margin:10px"
type="object" name="action_sync_contacts"
attrs="{'invisible': [('contact_synced', '=', True)]}"
/>
</group>
<group>
<button string="Sync Leads" class="btn-primary" style="margin:10px"
type="object" name="action_sync_leads"
attrs="{'invisible': [('lead_synced', '=', True)]}"/>
</group>
</group>
</page>
</xpath>
</field>
</record>
</odoo>
Loading…
Cancel
Save