diff --git a/developer_mode/README.rst b/developer_mode/README.rst new file mode 100644 index 000000000..cba11e830 --- /dev/null +++ b/developer_mode/README.rst @@ -0,0 +1,21 @@ +================================ + Automatic Developer Mode v12 +================================ + +Odoo Developers, Keep smiling for the below reasons: + + * Automatically Trigger Developer Mode. + * Separate Group for 'Odoo Developers'. + * Upgrade Modules Easily. + * Recently Upgraded Filter. + +Credits +------- +* `Nilmar Shereef < odoo@cybsosys.in >`__ +* `Saritha < odoo@cybsosys.in >`__ + + +Further information +=================== +HTML Description: ``__ + diff --git a/developer_mode/__init__.py b/developer_mode/__init__.py new file mode 100644 index 000000000..9e8b0b349 --- /dev/null +++ b/developer_mode/__init__.py @@ -0,0 +1,23 @@ +# -*- coding: utf-8 -*- +################################################################################### +# +# Cybrosys Technologies Pvt. Ltd. +# Copyright (C) 2018-TODAY Cybrosys Technologies(). +# Author: Cybrosys Techno Solutions () +# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# 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 for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +################################################################################### + +from . import controllers diff --git a/developer_mode/__manifest__.py b/developer_mode/__manifest__.py new file mode 100644 index 000000000..e0f6cf07e --- /dev/null +++ b/developer_mode/__manifest__.py @@ -0,0 +1,41 @@ +# -*- coding: utf-8 -*- +################################################################################### +# +# Cybrosys Technologies Pvt. Ltd. +# Copyright (C) 2019-TODAY Cybrosys Technologies(). +# Author: Cybrosys Techno Solutions () +# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# 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 for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +################################################################################### + +{ + 'name': "Automatic Developer Mode", + 'summary': """Automatically Activate Developer Mode & Running DB Name on Left Top""", + 'version': '12.0.1.0.0', + 'author': 'Cybrosys Techno Solutions', + 'website': "https://www.cybrosys.com", + 'company': 'Cybrosys Techno Solutions', + 'category': 'Extra Tools', + 'depends': ['base', 'web', 'base_setup'], + 'data': [ + 'security/security_data.xml', + 'views/developer_mode_view.xml', + # 'data/cybro_developer_data.xml', + ], + 'images': ['static/description/banner.jpg'], + 'license': 'AGPL-3', + 'installable': True, + 'auto_install': False, +} diff --git a/developer_mode/controllers/__init__.py b/developer_mode/controllers/__init__.py new file mode 100644 index 000000000..619523c05 --- /dev/null +++ b/developer_mode/controllers/__init__.py @@ -0,0 +1,23 @@ +# -*- coding: utf-8 -*- +################################################################################### +# +# Cybrosys Technologies Pvt. Ltd. +# Copyright (C) 2018-TODAY Cybrosys Technologies(). +# Author: Cybrosys Techno Solutions () +# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# 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 for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +################################################################################### + +from . import main diff --git a/developer_mode/controllers/main.py b/developer_mode/controllers/main.py new file mode 100644 index 000000000..d3b0d4ff9 --- /dev/null +++ b/developer_mode/controllers/main.py @@ -0,0 +1,64 @@ +# -*- coding: utf-8 -*- +################################################################################### +# +# Cybrosys Technologies Pvt. Ltd. +# Copyright (C) 2018-TODAY Cybrosys Technologies(). +# Author: Cybrosys Techno Solutions () +# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# 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 for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +################################################################################### + +import odoo +from odoo import http, _ +from odoo.http import request +from odoo.addons.web.controllers.main import Home, ensure_db + + +class AutoDeveloperMode(Home): + + @http.route('/web/login', type='http', auth="none") + def web_login(self, redirect=None, **kw): + """ Controller functions overrides for redirecting to developer mode if the logging user is admin or + 'Odoo Developer' group member """ + ensure_db() + request.params['login_success'] = False + if request.httprequest.method == 'GET' and redirect and request.session.uid: + return http.redirect_with_hash(redirect) + + if not request.uid: + request.uid = odoo.SUPERUSER_ID + + values = request.params.copy() + try: + values['databases'] = http.db_list() + except odoo.exceptions.AccessDenied: + values['databases'] = None + if request.httprequest.method == 'POST': + old_uid = request.uid + uid = request.session.authenticate(request.session.db, request.params['login'], request.params['password']) + print("uid",uid) + if uid is not False: + request.params['login_success'] = True + if not redirect: + odoo_technician = request.env.user.has_group('developer_mode.odoo_developer_group') + + if odoo_technician or request.uid == True: + redirect = '/web?debug' + else: + redirect = '/web' + return http.redirect_with_hash(redirect) + request.uid = old_uid + values['error'] = _("Wrong login/password") + return request.render('web.login', values) diff --git a/developer_mode/data/cybro_developer_data.xml b/developer_mode/data/cybro_developer_data.xml new file mode 100644 index 000000000..d62bb466a --- /dev/null +++ b/developer_mode/data/cybro_developer_data.xml @@ -0,0 +1,47 @@ + + + + + + + res.partner + cybro_partner_developer_mode + True + + + + Cybro Developer + +****************************************************** + + Your Odoo ERP Consultant! + +****************************************************** + +Cybrosys provides tremendous scope of Odoo customization where all kinds of business needs can be satisfied and could further enhance later if required with ease. Talk to us! + + + + + Kinfra Techno Park + Neospace + Kakkancherry + 673635 + 1 + +91 (0) 4943015006 + +91 (0) 4943015007 + info@cybrsys.com + https://www.cybrosys.com + + + + + + + + + + + + + diff --git a/developer_mode/doc/RELEASE_NOTES.md b/developer_mode/doc/RELEASE_NOTES.md new file mode 100644 index 000000000..1a52502a2 --- /dev/null +++ b/developer_mode/doc/RELEASE_NOTES.md @@ -0,0 +1,6 @@ +## Module + +#### 27.02.2019 +#### Version 12.0.1.0.0 +#### ADD +- Initial Commit diff --git a/developer_mode/security/security_data.xml b/developer_mode/security/security_data.xml new file mode 100644 index 000000000..0f27f8933 --- /dev/null +++ b/developer_mode/security/security_data.xml @@ -0,0 +1,12 @@ + + + + + + Odoo Developer + + + + + + diff --git a/developer_mode/static/description/banner.jpg b/developer_mode/static/description/banner.jpg new file mode 100644 index 000000000..0d748f903 Binary files /dev/null and b/developer_mode/static/description/banner.jpg differ diff --git a/developer_mode/static/description/cybro.jpg b/developer_mode/static/description/cybro.jpg new file mode 100644 index 000000000..28b019bab Binary files /dev/null and b/developer_mode/static/description/cybro.jpg differ diff --git a/developer_mode/static/description/cybro_logo.png b/developer_mode/static/description/cybro_logo.png new file mode 100644 index 000000000..bb309114c Binary files /dev/null and b/developer_mode/static/description/cybro_logo.png differ diff --git a/developer_mode/static/description/cybrosys-developer-mode-1.png b/developer_mode/static/description/cybrosys-developer-mode-1.png new file mode 100644 index 000000000..0162caa24 Binary files /dev/null and b/developer_mode/static/description/cybrosys-developer-mode-1.png differ diff --git a/developer_mode/static/description/cybrosys-developer-mode-2.jpg b/developer_mode/static/description/cybrosys-developer-mode-2.jpg new file mode 100644 index 000000000..2f4017465 Binary files /dev/null and b/developer_mode/static/description/cybrosys-developer-mode-2.jpg differ diff --git a/developer_mode/static/description/cybrosys-developer-mode-3.png b/developer_mode/static/description/cybrosys-developer-mode-3.png new file mode 100644 index 000000000..78e0a709d Binary files /dev/null and b/developer_mode/static/description/cybrosys-developer-mode-3.png differ diff --git a/developer_mode/static/description/cybrosys-developer-mode-4.png b/developer_mode/static/description/cybrosys-developer-mode-4.png new file mode 100644 index 000000000..4a4049cbf Binary files /dev/null and b/developer_mode/static/description/cybrosys-developer-mode-4.png differ diff --git a/developer_mode/static/description/cybrosys-developer-mode-5.png b/developer_mode/static/description/cybrosys-developer-mode-5.png new file mode 100644 index 000000000..1c105e07b Binary files /dev/null and b/developer_mode/static/description/cybrosys-developer-mode-5.png differ diff --git a/developer_mode/static/description/dev.jpg b/developer_mode/static/description/dev.jpg new file mode 100644 index 000000000..3cef12199 Binary files /dev/null and b/developer_mode/static/description/dev.jpg differ diff --git a/developer_mode/static/description/icon.png b/developer_mode/static/description/icon.png new file mode 100644 index 000000000..d31867890 Binary files /dev/null and b/developer_mode/static/description/icon.png differ diff --git a/developer_mode/static/description/index.html b/developer_mode/static/description/index.html new file mode 100644 index 000000000..c53542026 --- /dev/null +++ b/developer_mode/static/description/index.html @@ -0,0 +1,362 @@ +
+
+

+ Automatic Developer Mode +

+

+ Developers, Keep up your smile! +

+
+ Cybrosys Technologies +
+ +
+ cybrosys technologies +
+
+
+
+
+
+

+ Overview +

+

+ This module makes you free from activating developer mode operations repeatedly. + When you login, It will trigger the DEVELOPER MODE automatically. + It also helps in identification of running DB and make upgradation of modules easier. +

+

+ Configuration +

+

+ You have to enable 'Odoo Developer' group for respective users (Except Admin)in settings & + YOU HAVE TO RE-LOGIN AFTER THE MODULE INSTALLATION. +

+
+
+
+
+

+ Features +

+

+ + Automatically Trigger Developer Mode. +

+

+ + Separate Group for 'Odoo Developers'. +

+

+ + Showing Running DB On Left Top. +

+

+ + Upgrade Modules Easily. +

+
+
+ +
+
+

+ Screenshots +

+

+ + As shown here, Simply automate the developer mode. It helps developer in several perspective. +

+
+ +
+

+ + Shows the running DB on the left top. +

+
+ +
+

+ + Easily upgrade any module without going the module form. +

+
+ +
+

+ + Recent upgraded filter view in Apps. +

+
+ +
+
+
+ +
+
+ cybrosys technologies +
+
+
+
+

+ Our Services +

+
+ + + +
+ +
+ + + +
+

+ + Odoo Support +

+ +
+ +
+
+
+
+
+

+ Our Industries +

+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Trading + +

+

+ Easily procure and sell your products. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Manufacturing +

+

+ Plan, track and schedule your operations. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Restaurant +

+

+ Run your bar or restaurant methodical. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + POS +

+

+ Easy configuring and convivial selling. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + E-commerce & Website +

+

+ Mobile friendly, awe-inspiring product pages. +

+
+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Hotel Management +

+

+ An all-inclusive hotel management application. +

+
+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Education +

+

+ A Collaborative platform for educational management. +

+
+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Service Management +

+

+ Keep track of services and invoice accordingly. +

+
+
+
+
+
+
+ +
diff --git a/developer_mode/views/developer_mode_view.xml b/developer_mode/views/developer_mode_view.xml new file mode 100644 index 000000000..d9ade0cb4 --- /dev/null +++ b/developer_mode/views/developer_mode_view.xml @@ -0,0 +1,49 @@ + + + + + + ir.module.module.list.select + + ir.module.module + + + + + + + + + + + + + Modules Kanban + + ir.module.module + + + + + + + + + + + + + + + + + + + + diff --git a/developer_mode/views/ir_rule_view.xml b/developer_mode/views/ir_rule_view.xml new file mode 100644 index 000000000..fe7a181f7 --- /dev/null +++ b/developer_mode/views/ir_rule_view.xml @@ -0,0 +1,16 @@ + + + + + + ir.rule + + + + + + + + + + diff --git a/product_image_suggestion/README.rst b/product_image_suggestion/README.rst new file mode 100644 index 000000000..513c29627 --- /dev/null +++ b/product_image_suggestion/README.rst @@ -0,0 +1,55 @@ +============================ +Product Image Suggestion V12 +============================ + +This module shortlists you with the most relevant product images on Google Images that you are precisely looking for. + +Installation +============ + +- www.odoo.com/documentation/12.0/setup/install.html +- Install our custom addon + + +Configuration +============= + +* Before going to install, make sure "PIL" and "python-resize-image" python packages are installed on your server +* After installing the module, go to product form. Search product image from page "Image suggestion". +* Use image number limit for searching. +* Resize image if necessary.for install Resizeimage :pip3 install --user python-resize-image + + +Features +======== + +* A seamless and accurate search of product images from Google. +* Prioritizing the image suggestions via setting the desired search limit. +* Lock screen using keyboard shortcut 'CTRL+L'. +* Easy image editing; Can automatically resize the image to 1024x1204 px + + +License +======= +GNU LESSER GENERAL PUBLIC LICENSE, Version 3 (LGPLv3) +(http://www.gnu.org/licenses/agpl.html) + +Bug Tracker +=========== +Bugs are tracked on GitHub Issues. In case of trouble, please check there if your issue has already been reported. + +Credits +======= +* Cybrosys Techno Solutions + +Author +------ + +Developer: Akhilesh N S @ cybrosys, odoo@cybrosys.com + +Maintainer +---------- + +This module is maintained by Cybrosys Technologies. + +For support and more information, please visit https://www.cybrosys.com. diff --git a/product_image_suggestion/__init__.py b/product_image_suggestion/__init__.py new file mode 100644 index 000000000..a0fdc10fe --- /dev/null +++ b/product_image_suggestion/__init__.py @@ -0,0 +1,2 @@ +# -*- coding: utf-8 -*- +from . import models diff --git a/product_image_suggestion/__manifest__.py b/product_image_suggestion/__manifest__.py new file mode 100644 index 000000000..87993e47c --- /dev/null +++ b/product_image_suggestion/__manifest__.py @@ -0,0 +1,43 @@ +# -*- coding: utf-8 -*- + +############################################################################## +# +# Cybrosys Technologies Pvt. Ltd. +# Copyright (C) 2019-TODAY Cybrosys Technologies(). +# Author: Akhilesh N S +# you can modify it under the terms of the GNU LESSER +# GENERAL PUBLIC LICENSE (LGPL v3), Version 3. +# +# It is forbidden to publish, distribute, sublicense, or sell copies +# of the Software or modified copies of the Software. +# +# 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. +# +# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE +# GENERAL PUBLIC LICENSE (LGPL v3) along with this program. +# If not, see . +# +############################################################################## +{ + 'name': 'Product Image Suggestion', + 'version': '12.0.1.0.0', + 'summary': 'Suggest product images from google search', + 'category': 'Tools', + 'author': 'Cybrosys Techno Solutions', + 'company': 'Cybrosys Techno Solutions', + 'maintainer': 'Cybrosys Techno Solutions', + 'depends': ['base', 'product'], + 'external_dependencies': {'python': ['base64', 'PIL','resizeimage']}, + 'website': 'https://www.cybrosys.com', + 'data': [ + 'views/product_template.xml', + ], + 'images': ['static/description/banner.gif'], + 'license': 'LGPL-3', + 'installable': True, + 'auto_install': False, + 'application': False, +} diff --git a/product_image_suggestion/doc/RELEASE_NOTES.md b/product_image_suggestion/doc/RELEASE_NOTES.md new file mode 100644 index 000000000..c3d306490 --- /dev/null +++ b/product_image_suggestion/doc/RELEASE_NOTES.md @@ -0,0 +1,8 @@ +## Module + +#### 27.02.2019 +#### Version 12.0.1.0.0 +#### ADD + +- Initial commit + diff --git a/product_image_suggestion/models/__init__.py b/product_image_suggestion/models/__init__.py new file mode 100644 index 000000000..07bfe9114 --- /dev/null +++ b/product_image_suggestion/models/__init__.py @@ -0,0 +1,3 @@ +# -*- coding: utf-8 -*- +from . import product_image_suggestion +from . import google_images_download diff --git a/product_image_suggestion/models/google_images_download.py b/product_image_suggestion/models/google_images_download.py new file mode 100644 index 000000000..3726fb662 --- /dev/null +++ b/product_image_suggestion/models/google_images_download.py @@ -0,0 +1,622 @@ +#!/usr/bin/env python +# In[ ]: +# coding: utf-8 + +###### Searching and Downloading Google Images to the local disk ###### + +# Import Libraries +import sys +import logging +_logger = logging.getLogger(__name__) + +version = (3, 0) +cur_version = sys.version_info +if cur_version >= version: # If the Current Version of Python is 3.0 or above + import urllib.request + from urllib.request import Request, urlopen + from urllib.request import URLError, HTTPError + from urllib.parse import quote + import http.client + from http.client import IncompleteRead + http.client._MAXHEADERS = 1000 +else: # If the Current Version of Python is 2.x + import urllib2 + from urllib2 import Request, urlopen + from urllib2 import URLError, HTTPError + from urllib import quote + import httplib + from httplib import IncompleteRead + httplib._MAXHEADERS = 1000 +import time # Importing the time library to check the time of code execution +import os +import argparse +import ssl +import datetime +import json +import re +import codecs +import socket + +args_list = ["keywords", "keywords_from_file", "prefix_keywords", "suffix_keywords", + "limit", "format", "color", "color_type", "usage_rights", "size", + "exact_size", "aspect_ratio", "type", "time", "time_range", "delay", "url", "single_image", + "output_directory", "image_directory", "no_directory", "proxy", "similar_images", "specific_site", + "print_urls", "print_size", "print_paths", "metadata", "extract_metadata", "socket_timeout", + "thumbnail", "language", "prefix", "chromedriver", "related_images", "safe_search", "no_numbering", + "offset", "no_download"] + + +def user_input(): + config = argparse.ArgumentParser() + config.add_argument('-cf', '--config_file', help='config file name', default='', type=str, required=False) + config_file_check = config.parse_known_args() + object_check = vars(config_file_check[0]) + + if object_check['config_file'] != '': + records = [] + json_file = json.load(open(config_file_check[0].config_file)) + for record in range(0,len(json_file['Records'])): + arguments = {} + for i in args_list: + arguments[i] = None + for key, value in json_file['Records'][record].items(): + arguments[key] = value + records.append(arguments) + records_count = len(records) + else: + # Taking command line arguments from users + parser = argparse.ArgumentParser() + parser.add_argument('-k', '--keywords', help='delimited list input', type=str, required=False) + parser.add_argument('-kf', '--keywords_from_file', help='extract list of keywords from a text file', type=str, required=False) + parser.add_argument('-sk', '--suffix_keywords', help='comma separated additional words added after to main keyword', type=str, required=False) + parser.add_argument('-pk', '--prefix_keywords', help='comma separated additional words added before main keyword', type=str, required=False) + parser.add_argument('-l', '--limit', help='delimited list input', type=str, required=False) + parser.add_argument('-f', '--format', help='download images with specific format', type=str, required=False, + choices=['jpg', 'gif', 'png', 'bmp', 'svg', 'webp', 'ico']) + parser.add_argument('-u', '--url', help='search with google image URL', type=str, required=False) + parser.add_argument('-x', '--single_image', help='downloading a single image from URL', type=str, required=False) + parser.add_argument('-o', '--output_directory', help='download images in a specific main directory', type=str, required=False) + parser.add_argument('-i', '--image_directory', help='download images in a specific sub-directory', type=str, required=False) + parser.add_argument('-n', '--no_directory', default=False, help='download images in the main directory but no sub-directory', action="store_true") + parser.add_argument('-d', '--delay', help='delay in seconds to wait between downloading two images', type=int, required=False) + parser.add_argument('-co', '--color', help='filter on color', type=str, required=False, + choices=['red', 'orange', 'yellow', 'green', 'teal', 'blue', 'purple', 'pink', 'white', 'gray', 'black', 'brown']) + parser.add_argument('-ct', '--color_type', help='filter on color', type=str, required=False, + choices=['full-color', 'black-and-white', 'transparent']) + parser.add_argument('-r', '--usage_rights', help='usage rights', type=str, required=False, + choices=['labeled-for-reuse-with-modifications','labeled-for-reuse','labeled-for-noncommercial-reuse-with-modification','labeled-for-nocommercial-reuse']) + parser.add_argument('-s', '--size', help='image size', type=str, required=False, + choices=['large','medium','icon','>400*300','>640*480','>800*600','>1024*768','>2MP','>4MP','>6MP','>8MP','>10MP','>12MP','>15MP','>20MP','>40MP','>70MP']) + parser.add_argument('-es', '--exact_size', help='exact image resolution "WIDTH,HEIGHT"', type=str, required=False) + parser.add_argument('-t', '--type', help='image type', type=str, required=False, + choices=['face','photo','clipart','line-drawing','animated']) + parser.add_argument('-w', '--time', help='image age', type=str, required=False, + choices=['past-24-hours','past-7-days']) + parser.add_argument('-wr', '--time_range', help='time range for the age of the image. should be in the format {"time_min":"MM/DD/YYYY","time_max":"MM/DD/YYYY"}', type=str, required=False) + parser.add_argument('-a', '--aspect_ratio', help='comma separated additional words added to keywords', type=str, required=False, + choices=['tall', 'square', 'wide', 'panoramic']) + parser.add_argument('-si', '--similar_images', help='downloads images very similar to the image URL you provide', type=str, required=False) + parser.add_argument('-ss', '--specific_site', help='downloads images that are indexed from a specific website', type=str, required=False) + parser.add_argument('-p', '--print_urls', default=False, help="Print the URLs of the images", action="store_true") + parser.add_argument('-ps', '--print_size', default=False, help="Print the size of the images on disk", action="store_true") + parser.add_argument('-pp', '--print_paths', default=False, help="Prints the list of absolute paths of the images",action="store_true") + parser.add_argument('-m', '--metadata', default=False, help="Print the metadata of the image", action="store_true") + parser.add_argument('-e', '--extract_metadata', default=False, help="Dumps all the logs into a text file", action="store_true") + parser.add_argument('-st', '--socket_timeout', default=False, help="Connection timeout waiting for the image to download", type=float) + parser.add_argument('-th', '--thumbnail', default=False, help="Downloads image thumbnail along with the actual image", action="store_true") + parser.add_argument('-la', '--language', default=False, help="Defines the language filter. The search results are authomatically returned in that language", type=str, required=False, + choices=['Arabic','Chinese (Simplified)','Chinese (Traditional)','Czech','Danish','Dutch','English','Estonian','Finnish','French','German','Greek','Hebrew','Hungarian','Icelandic','Italian','Japanese','Korean','Latvian','Lithuanian','Norwegian','Portuguese','Polish','Romanian','Russian','Spanish','Swedish','Turkish']) + parser.add_argument('-pr', '--prefix', default=False, help="A word that you would want to prefix in front of each image name", type=str, required=False) + parser.add_argument('-px', '--proxy', help='specify a proxy address and port', type=str, required=False) + parser.add_argument('-cd', '--chromedriver', help='specify the path to chromedriver executable in your local machine', type=str, required=False) + parser.add_argument('-ri', '--related_images', default=False, help="Downloads images that are similar to the keyword provided", action="store_true") + parser.add_argument('-sa', '--safe_search', default=False, help="Turns on the safe search filter while searching for images", action="store_true") + parser.add_argument('-nn', '--no_numbering', default=False, help="Allows you to exclude the default numbering of images", action="store_true") + parser.add_argument('-of', '--offset', help="Where to start in the fetched links", type=str, required=False) + parser.add_argument('-nd', '--no_download', default=False, help="Prints the URLs of the images and/or thumbnails without downloading them", action="store_true") + + args = parser.parse_args() + arguments = vars(args) + records = [] + records.append(arguments) + return records + + +class googleimagesdownload: + def __init__(self): + pass + + # Downloading entire Web Document (Raw Page Content) + def download_page(self,url): + version = (3, 0) + cur_version = sys.version_info + if cur_version >= version: # If the Current Version of Python is 3.0 or above + try: + headers = {} + headers['User-Agent'] = "Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/41.0.2228.0 Safari/537.36" + req = urllib.request.Request(url, headers=headers) + resp = urllib.request.urlopen(req) + respData = str(resp.read()) + return respData + except Exception as e: + _logger.exception("Could not open URL. Please check your internet connection and/or ssl settings") + else: # If the Current Version of Python is 2.x + try: + headers = {} + headers['User-Agent'] = "Mozilla/5.0 (X11; Linux i686) AppleWebKit/537.17 (KHTML, like Gecko) Chrome/24.0.1312.27 Safari/537.17" + req = urllib2.Request(url, headers=headers) + try: + response = urllib2.urlopen(req) + except URLError: # Handling SSL certificate failed + context = ssl._create_unverified_context() + response = urlopen(req, context=context) + page = response.read() + return page + except: + _logger.exception("Could not open URL. Please check your internet connection and/or ssl settings") + return "Page Not found" + + + # Download Page for more than 100 images + def download_extended_page(self,url,chromedriver): + from selenium import webdriver + from selenium.webdriver.common.keys import Keys + if sys.version_info[0] < 3: + reload(sys) + sys.setdefaultencoding('utf8') + options = webdriver.ChromeOptions() + options.add_argument('--no-sandbox') + options.add_argument("--headless") + + try: + browser = webdriver.Chrome(chromedriver, chrome_options=options) + except Exception as e: + _logger.exception("Looks like we cannot locate the path the 'chromedriver' (use the '--chromedriver' " + "argument to specify the path to the executable.) or google chrome browser is not " + "installed on your machine (exception: %s)" % e) + + sys.exit() + browser.set_window_size(1024, 768) + + # Open the link + browser.get(url) + time.sleep(1) + _logger.warning("Getting you a lot of images. This may take a few moments...") + + element = browser.find_element_by_tag_name("body") + # Scroll down + for i in range(30): + element.send_keys(Keys.PAGE_DOWN) + time.sleep(0.3) + + try: + browser.find_element_by_id("smb").click() + for i in range(50): + element.send_keys(Keys.PAGE_DOWN) + time.sleep(0.3) # bot id protection + except: + for i in range(10): + element.send_keys(Keys.PAGE_DOWN) + time.sleep(0.3) # bot id protection + + _logger.warning("Reached end of Page.") + + time.sleep(0.5) + + source = browser.page_source #page source + #close the browser + browser.close() + + return source + + + #Correcting the escape characters for python2 + def replace_with_byte(self,match): + return chr(int(match.group(0)[1:], 8)) + + def repair(self,brokenjson): + invalid_escape = re.compile(r'\\[0-7]{1,3}') # up to 3 digits for byte values up to FF + return invalid_escape.sub(self.replace_with_byte, brokenjson) + + + # Finding 'Next Image' from the given raw page + def get_next_tab(self,s): + start_line = s.find('class="dtviD"') + if start_line == -1: # If no links are found then give an error! + end_quote = 0 + link = "no_tabs" + return link,'',end_quote + else: + start_line = s.find('class="dtviD"') + start_content = s.find('href="', start_line + 1) + end_content = s.find('">', start_content + 1) + url_item = "https://www.google.com" + str(s[start_content+6:end_content]) + url_item = url_item.replace('&', '&') + + start_line_2 = s.find('class="dtviD"') + start_content_2 = s.find(':', start_line_2 + 1) + end_content_2 = s.find('"', start_content_2 + 1) + url_item_name = str(s[start_content_2 + 1:end_content_2]) + + return url_item,url_item_name,end_content + + + # Getting all links with the help of '_images_get_next_image' + def get_all_tabs(self,page): + tabs = {} + while True: + item,item_name,end_content = self.get_next_tab(page) + if item == "no_tabs": + break + else: + tabs[item_name] = item # Append all the links in the list named 'Links' + time.sleep(0.1) # Timer could be used to slow down the request for image downloads + page = page[end_content:] + return tabs + + + #Format the object in readable format + def format_object(self,object): + formatted_object = {} + formatted_object['image_format'] = object['ity'] + formatted_object['image_height'] = object['oh'] + formatted_object['image_width'] = object['ow'] + formatted_object['image_link'] = object['ou'] + formatted_object['image_description'] = object['pt'] + formatted_object['image_host'] = object['rh'] + formatted_object['image_source'] = object['ru'] + formatted_object['image_thumbnail_url'] = object['tu'] + return formatted_object + + + def similar_images(self,similar_images): + version = (3, 0) + cur_version = sys.version_info + if cur_version >= version: # If the Current Version of Python is 3.0 or above + try: + searchUrl = 'https://www.google.com/searchbyimage?site=search&sa=X&image_url=' + similar_images + headers = {} + headers['User-Agent'] = "Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/41.0.2228.0 Safari/537.36" + + req1 = urllib.request.Request(searchUrl, headers=headers) + resp1 = urllib.request.urlopen(req1) + content = str(resp1.read()) + l1 = content.find('AMhZZ') + l2 = content.find('&', l1) + urll = content[l1:l2] + + newurl = "https://www.google.com/search?tbs=sbi:" + urll + "&site=search&sa=X" + req2 = urllib.request.Request(newurl, headers=headers) + resp2 = urllib.request.urlopen(req2) + + l3 = content.find('/search?sa=X&q=') + l4 = content.find(';', l3 + 19) + urll2 = content[l3 + 19:l4] + return urll2 + except: + return "Cloud not connect to Google Images endpoint" + else: # If the Current Version of Python is 2.x + try: + searchUrl = 'https://www.google.com/searchbyimage?site=search&sa=X&image_url=' + similar_images + headers = {} + headers['User-Agent'] = "Mozilla/5.0 (X11; Linux i686) AppleWebKit/537.17 (KHTML, like Gecko) Chrome/24.0.1312.27 Safari/537.17" + + req1 = urllib2.Request(searchUrl, headers=headers) + resp1 = urllib2.urlopen(req1) + content = str(resp1.read()) + l1 = content.find('AMhZZ') + l2 = content.find('&', l1) + urll = content[l1:l2] + + newurl = "https://www.google.com/search?tbs=sbi:" + urll + "&site=search&sa=X" + + req2 = urllib2.Request(newurl, headers=headers) + resp2 = urllib2.urlopen(req2) + + l3 = content.find('/search?sa=X&q=') + l4 = content.find(';', l3 + 19) + urll2 = content[l3 + 19:l4] + return(urll2) + except: + return "Cloud not connect to Google Images endpoint" + + #Building URL parameters + def build_url_parameters(self,arguments): + if arguments['language']: + lang = "&lr=" + lang_param = {"Arabic":"lang_ar","Chinese (Simplified)":"lang_zh-CN","Chinese (Traditional)":"lang_zh-TW","Czech":"lang_cs","Danish":"lang_da","Dutch":"lang_nl","English":"lang_en","Estonian":"lang_et","Finnish":"lang_fi","French":"lang_fr","German":"lang_de","Greek":"lang_el","Hebrew":"lang_iw ","Hungarian":"lang_hu","Icelandic":"lang_is","Italian":"lang_it","Japanese":"lang_ja","Korean":"lang_ko","Latvian":"lang_lv","Lithuanian":"lang_lt","Norwegian":"lang_no","Portuguese":"lang_pt","Polish":"lang_pl","Romanian":"lang_ro","Russian":"lang_ru","Spanish":"lang_es","Swedish":"lang_sv","Turkish":"lang_tr"} + lang_url = lang+lang_param[arguments['language']] + else: + lang_url = '' + + if arguments['time_range']: + json_acceptable_string = arguments['time_range'].replace("'", "\"") + d = json.loads(json_acceptable_string) + time_range = ',cdr:1,cd_min:' + d['time_min'] + ',cd_max:' + d['time_max'] + else: + time_range = '' + + if arguments['exact_size']: + size_array = [x.strip() for x in arguments['exact_size'].split(',')] + exact_size = ",isz:ex,iszw:" + str(size_array[0]) + ",iszh:" + str(size_array[1]) + else: + exact_size = '' + + built_url = "&tbs=" + counter = 0 + params = {'color':[arguments['color'],{'red':'ic:specific,isc:red', 'orange':'ic:specific,isc:orange', 'yellow':'ic:specific,isc:yellow', 'green':'ic:specific,isc:green', 'teal':'ic:specific,isc:teel', 'blue':'ic:specific,isc:blue', 'purple':'ic:specific,isc:purple', 'pink':'ic:specific,isc:pink', 'white':'ic:specific,isc:white', 'gray':'ic:specific,isc:gray', 'black':'ic:specific,isc:black', 'brown':'ic:specific,isc:brown'}], + 'color_type':[arguments['color_type'],{'full-color':'ic:color', 'black-and-white':'ic:gray','transparent':'ic:trans'}], + 'usage_rights':[arguments['usage_rights'],{'labeled-for-reuse-with-modifications':'sur:fmc','labeled-for-reuse':'sur:fc','labeled-for-noncommercial-reuse-with-modification':'sur:fm','labeled-for-nocommercial-reuse':'sur:f'}], + 'size':[arguments['size'],{'large':'isz:l','medium':'isz:m','icon':'isz:i','>400*300':'isz:lt,islt:qsvga','>640*480':'isz:lt,islt:vga','>800*600':'isz:lt,islt:svga','>1024*768':'visz:lt,islt:xga','>2MP':'isz:lt,islt:2mp','>4MP':'isz:lt,islt:4mp','>6MP':'isz:lt,islt:6mp','>8MP':'isz:lt,islt:8mp','>10MP':'isz:lt,islt:10mp','>12MP':'isz:lt,islt:12mp','>15MP':'isz:lt,islt:15mp','>20MP':'isz:lt,islt:20mp','>40MP':'isz:lt,islt:40mp','>70MP':'isz:lt,islt:70mp'}], + 'type':[arguments['type'],{'face':'itp:face','photo':'itp:photo','clipart':'itp:clipart','line-drawing':'itp:lineart','animated':'itp:animated'}], + 'time':[arguments['time'],{'past-24-hours':'qdr:d','past-7-days':'qdr:w'}], + 'aspect_ratio':[arguments['aspect_ratio'],{'tall':'iar:t','square':'iar:s','wide':'iar:w','panoramic':'iar:xw'}], + 'format':[arguments['format'],{'jpg':'ift:jpg','gif':'ift:gif','png':'ift:png','bmp':'ift:bmp','svg':'ift:svg','webp':'webp','ico':'ift:ico'}]} + for key, value in params.items(): + if value[0] is not None: + ext_param = value[1][value[0]] + # counter will tell if it is first param added or not + if counter == 0: + # add it to the built url + built_url = built_url + ext_param + counter += 1 + else: + built_url = built_url + ',' + ext_param + counter += 1 + built_url = lang_url+built_url+exact_size+time_range + return built_url + + + #building main search URL + def build_search_url(self,search_term,params,url,similar_images,specific_site,safe_search): + #check safe_search + safe_search_string = "&safe=active" + # check the args and choose the URL + if url: + url = url + elif similar_images: + _logger.exception(similar_images) + keywordem = self.similar_images(similar_images) + url = 'https://www.google.com/search?q=' + keywordem + '&espv=2&biw=1366&bih=667&site=webhp&source=lnms&tbm=isch&sa=X&ei=XosDVaCXD8TasATItgE&ved=0CAcQ_AUoAg' + elif specific_site: + url = 'https://www.google.com/search?q=' + quote( + search_term) + '&as_sitesearch=' + specific_site + '&espv=2&biw=1366&bih=667&site=webhp&source=lnms&tbm=isch' + params + '&sa=X&ei=XosDVaCXD8TasATItgE&ved=0CAcQ_AUoAg' + else: + url = 'https://www.google.com/search?q=' + quote( + search_term) + '&espv=2&biw=1366&bih=667&site=webhp&source=lnms&tbm=isch' + params + '&sa=X&ei=XosDVaCXD8TasATItgE&ved=0CAcQ_AUoAg' + + #safe search check + if safe_search: + url = url + safe_search_string + + + return url + + + #measures the file size + def file_size(self,file_path): + if os.path.isfile(file_path): + file_info = os.stat(file_path) + size = file_info.st_size + for x in ['bytes', 'KB', 'MB', 'GB', 'TB']: + if size < 1024.0: + return "%3.1f %s" % (size, x) + size /= 1024.0 + return size + + #keywords from file + def keywords_from_file(self,file_name): + search_keyword = [] + with codecs.open(file_name, 'r', encoding='utf-8-sig') as f: + if '.csv' in file_name: + for line in f: + if line in ['\n', '\r\n']: + pass + else: + search_keyword.append(line.replace('\n', '').replace('\r', '')) + elif '.txt' in file_name: + for line in f: + if line in ['\n', '\r\n']: + pass + else: + search_keyword.append(line.replace('\n', '').replace('\r', '')) + else: + _logger.warning("Invalid file type: Valid file types are either .txt or .csv \n" + "exiting...") + sys.exit() + return search_keyword + + + # Download Images + def download_image(self,image_url,image_format,count,print_urls,socket_timeout,prefix,print_size,no_numbering,no_download): + if not prefix: + prefix = '' + if print_urls or no_download: + _logger.info("Image URL: " + image_url) + if no_download: + return "success","Printed url without downloading",None,None + image_data = None + try: + req = Request(image_url, headers={ + "User-Agent": "Mozilla/5.0 (X11; Linux i686) AppleWebKit/537.17 (KHTML, like Gecko) Chrome/24.0.1312.27 Safari/537.17"}) + try: + # timeout time to download an image + if socket_timeout: + timeout = float(socket_timeout) + else: + timeout = 10 + + response = urlopen(req, None, timeout) + image_data = response.read() + response.close() + download_status = 'success' + download_message = "Completed Image ====> " + prefix + str(count) + + except UnicodeEncodeError as e: + download_status = 'fail' + download_message = "UnicodeEncodeError on an image...trying next one..." + " Error: " + str(e) + + except URLError as e: + download_status = 'fail' + download_message = "URLError on an image...trying next one..." + " Error: " + str(e) + + except HTTPError as e: # If there is any HTTPError + download_status = 'fail' + download_message = "HTTPError on an image...trying next one..." + " Error: " + str(e) + + except URLError as e: + download_status = 'fail' + download_message = "URLError on an image...trying next one..." + " Error: " + str(e) + + except ssl.CertificateError as e: + download_status = 'fail' + download_message = "CertificateError on an image...trying next one..." + " Error: " + str(e) + + except IOError as e: # If there is any IOError + download_status = 'fail' + download_message = "IOError on an image...trying next one..." + " Error: " + str(e) + + except IncompleteRead as e: + download_status = 'fail' + download_message = "IncompleteReadError on an image...trying next one..." + " Error: " + str(e) + + return download_status,download_message, image_data + + + # Finding 'Next Image' from the given raw page + def _get_next_item(self,s): + start_line = s.find('rg_meta notranslate') + if start_line == -1: # If no links are found then give an error! + end_quote = 0 + link = "no_links" + return link, end_quote + else: + start_line = s.find('class="rg_meta notranslate">') + start_object = s.find('{', start_line + 1) + end_object = s.find('', start_object + 1) + object_raw = str(s[start_object:end_object]) + #remove escape characters based on python version + version = (3, 0) + cur_version = sys.version_info + if cur_version >= version: #python3 + try: + object_decode = bytes(object_raw, "utf-8").decode("unicode_escape") + final_object = json.loads(object_decode) + except: + final_object = "" + else: #python2 + try: + final_object = (json.loads(self.repair(object_raw))) + except: + final_object = "" + return final_object, end_object + + + # Getting all links with the help of '_images_get_next_image' + # def _get_all_items(self,page,main_directory,dir_name,limit,arguments): + def _get_all_items(self,page,limit,arguments): + image_dic = [] + errorCount = 0 + i = 0 + count = 1 + while count < limit+1: + object, end_content = self._get_next_item(page) + if object == "no_links": + break + elif object == "": + page = page[end_content:] + elif arguments['offset'] and count < int(arguments['offset']): + count += 1 + page = page[end_content:] + else: + #format the item for readability + object = self.format_object(object) + # if arguments['metadata']: + # _logger.info("\nImage Metadata: " + str(object)) + #download the images + download_status,download_message,image_data = self.download_image(object['image_link'],object['image_format'],count,arguments['print_urls'],arguments['socket_timeout'],arguments['prefix'],arguments['print_size'],arguments['no_numbering'],arguments['no_download']) + _logger.warning(download_message) + if download_status == "success": + count += 1 + image_dic.append(image_data) + else: + errorCount += 1 + page = page[end_content:] + i += 1 + if count < limit: + _logger.warning("\n\nUnfortunately all " + str( + limit) + " could not be downloaded because some images were not downloadable. " + str( + count-1) + " is all we got for this search filter!") + return errorCount,image_dic + + # Bulk Download + def download(self,arguments): + + #for input coming from other python files + if __name__ != "__main__": + for arg in args_list: + if arg not in arguments: + arguments[arg] = None + + ######Initialization and Validation of user arguments + if arguments['keywords']: + search_keyword = [str(item) for item in arguments['keywords'].split(',')] + + if arguments['keywords_from_file']: + search_keyword = self.keywords_from_file(arguments['keywords_from_file']) + + # both time and time range should not be allowed in the same query + if arguments['time'] and arguments['time_range']: + raise ValueError('Either time or time range should be used in a query. Both cannot be used at the same time.') + + # both time and time range should not be allowed in the same query + if arguments['size'] and arguments['exact_size']: + raise ValueError('Either "size" or "exact_size" should be used in a query. Both cannot be used at the same time.') + + # both image directory and no image directory should not be allowed in the same query + if arguments['image_directory'] and arguments['no_directory']: + raise ValueError('You can either specify image directory or specify no image directory, not both!') + + # Additional words added to keywords + if arguments['suffix_keywords']: + suffix_keywords = [" " + str(sk) for sk in arguments['suffix_keywords'].split(',')] + else: + suffix_keywords = [''] + + # Additional words added to keywords + if arguments['prefix_keywords']: + prefix_keywords = [str(sk) + " " for sk in arguments['prefix_keywords'].split(',')] + else: + prefix_keywords = [''] + + # Setting limit on number of images to be downloaded + if arguments['limit']: + limit = int(arguments['limit']) + else: + limit = 100 + image_dic = [] + for pky in prefix_keywords: + for sky in suffix_keywords: # 1.for every suffix keywords + i = 0 + while i < len(search_keyword): # 2.for every main keyword + iteration = "\n" + "Item no.: " + str(i + 1) + " -->" + " Item name = " + str(pky) + str(search_keyword[i] + str(sky)) + _logger.info(iteration) + _logger.info("Evaluating...") + + search_term = pky + search_keyword[i] + sky + params = self.build_url_parameters(arguments) #building URL with params + url = self.build_search_url(search_term,params,arguments['url'],arguments['similar_images'],arguments['specific_site'],arguments['safe_search']) #building main search url + if limit < 101: + raw_html = self.download_page(url) # download page + else: + raw_html = self.download_extended_page(url,arguments['chromedriver']) + if arguments['no_download']: + _logger.info("Starting to Print Image URLS") + else: + _logger.info("Starting Download...") + errorCount,image_dic = self._get_all_items(raw_html,limit,arguments) #get all image items and download images + i += 1 + _logger.warning("\nErrors: " + str(errorCount) + "\n") + + return image_dic + diff --git a/product_image_suggestion/models/product_image_suggestion.py b/product_image_suggestion/models/product_image_suggestion.py new file mode 100644 index 000000000..a80bd2bd9 --- /dev/null +++ b/product_image_suggestion/models/product_image_suggestion.py @@ -0,0 +1,86 @@ +# -*- coding: utf-8 -*- +import base64 +import logging +import os +from PIL import Image +import tempfile +from resizeimage import resizeimage +from . import google_images_download +from odoo import models, fields, api, _ +from odoo.exceptions import UserError, Warning + + +_logger = logging.getLogger(__name__) + + +class ProductImageSelection(models.TransientModel): + _name = "product.image.suggestion" + + image = fields.Binary('Image', attachment=True) + product_tmpl_id = fields.Many2one('product.template', 'Related Product', copy=True) + + @api.multi + def action_set_image(self): + """Set product images from suggested images""" + self_image = self.image + if self_image: + self.product_tmpl_id.image = self_image + + +class ProductTemplate(models.Model): + _inherit = 'product.template' + + def get_search_string(self): + for prod in self: + prod.search_field = prod.name + + search_image_ids = fields.One2many('product.image.suggestion', 'product_tmpl_id', string='Images', readonly=True) + search_field = fields.Char('Search Text', compute=get_search_string, readonly=False, store=True) + image_limit = fields.Integer('Limit', default=5) + resize_image = fields.Boolean('Resize Image', default=True) + + @api.onchange('image_limit') + def war_image_limit(self): + if self.image_limit > 10: + raise Warning(_('This may slow down image search..!! !')) + + @api.multi + def search_images_button(self): + """Clear search images and add new search""" + self.search_image_ids = [[5,0,0,]] + if self.image_limit > 10: + _logger.warning("High limit number slow down the image searches") + try: + response = google_images_download.googleimagesdownload() + query_string = self.search_field.replace(" ", "_").replace(",", "_") + arguments = {"keywords": query_string, "limit": self.image_limit, "print_urls": False, 'safe_search': True} + image_datas = response.download(arguments) # passing the arguments to the function + except AttributeError: + raise UserError(_('No internet connection available or Something wrong !')) + if image_datas: + for im in image_datas: + temp_name = '' + try: + if self.resize_image: + temp_file, temp_name = tempfile.mkstemp(suffix='.png') + file = open(temp_name, "wb") + file.write(im) + file.close() + img = Image.open(temp_name) + img = resizeimage.resize_contain(img, [1024, 1024]) + img.save(temp_name, img.format) + with open(temp_name, "rb") as image_file: + binary_image = base64.b64encode(image_file.read()) + else: + b = bytearray(im) + binary_image = base64.b64encode(b) + vals = dict(image=binary_image, product_tmpl_id=self.id) + self.env['product.image.suggestion'].create(vals) + + if self.resize_image: + os.remove(temp_name) + except: + _logger.exception(_("failed to display in page")) + continue + else: + raise UserError(_('No image suggestions for this image')) diff --git a/product_image_suggestion/static/description/banner.gif b/product_image_suggestion/static/description/banner.gif new file mode 100644 index 000000000..cf2db96c3 Binary files /dev/null and b/product_image_suggestion/static/description/banner.gif differ diff --git a/product_image_suggestion/static/description/cybro_logo.png b/product_image_suggestion/static/description/cybro_logo.png new file mode 100644 index 000000000..bb309114c Binary files /dev/null and b/product_image_suggestion/static/description/cybro_logo.png differ diff --git a/product_image_suggestion/static/description/icon.png b/product_image_suggestion/static/description/icon.png new file mode 100644 index 000000000..a3e41dbc1 Binary files /dev/null and b/product_image_suggestion/static/description/icon.png differ diff --git a/product_image_suggestion/static/description/index.html b/product_image_suggestion/static/description/index.html new file mode 100644 index 000000000..15e91e000 --- /dev/null +++ b/product_image_suggestion/static/description/index.html @@ -0,0 +1,336 @@ + +
+
+

+ Product Image Suggestion +

+

+ Set Product Image by Google Search +

+
+ Cybrosys Technologies +
+ +
+ cybrosys technologies
+
+
+
+ +
+
+

+ Overview +

+

+ The Product Image Suggestion shortlists you with the most relevant product images on Google Images that you are precisely looking for. The Product Image Suggestion can help you get the best possible images via setting the desired limit to search, making the decisions faster and flawless. +

+

+ Configuration +

+

+ Before going to install, make sure "PIL" and "python-resize-image" python packages are installed on your server. After installing the module, go to product form. Search product image from page "Image suggestion". Use image number limit for searching. Resize image if necessary. +

+
+
+ +
+
+

+ Features +

+

+ + A seamless and accurate search of product images from Google +

+

+ + Prioritizing the image suggestions via setting the desired search limit +

+

+ + Easy image editing; Can automatically resize the image to 1024x1204 px +

+
+
+ +
+
+

+ Screenshots +

+

+
+ Product Form.
+

+
+ +
+
+
+
+
+ cybrosys technologies +
+
+ +
+
+

+ Our Services +

+
+ + + +
+ +
+ + + +
+

+ + Odoo Support +

+ +
+ +
+
+
+
+
+

+ Our Industries +

+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Trading + +

+

+ Easily procure and sell your products. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Manufacturing +

+

+ Plan, track and schedule your operations. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Restaurant +

+

+ Run your bar or restaurant methodical. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + POS +

+

+ Easy configuring and convivial selling. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + E-commerce & Website +

+

+ Mobile friendly, awe-inspiring product pages. +

+
+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Hotel Management +

+

+ An all-inclusive hotel management application. +

+
+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Education +

+

+ A Collaborative platform for educational management. +

+
+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Service Management +

+

+ Keep track of services and invoice accordingly. +

+
+
+
+
+
+
+ +
diff --git a/product_image_suggestion/static/description/product_form.jpg b/product_image_suggestion/static/description/product_form.jpg new file mode 100644 index 000000000..9a5f541af Binary files /dev/null and b/product_image_suggestion/static/description/product_form.jpg differ diff --git a/product_image_suggestion/views/product_template.xml b/product_image_suggestion/views/product_template.xml new file mode 100644 index 000000000..49ef85567 --- /dev/null +++ b/product_image_suggestion/views/product_template.xml @@ -0,0 +1,50 @@ + + + + + product.template.product.website.form + product.template + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +