diff --git a/gmail/.prettierrc b/gmail/.prettierrc index 7c8e25071..c219035ac 100644 --- a/gmail/.prettierrc +++ b/gmail/.prettierrc @@ -2,6 +2,6 @@ "semi": true, "trailingComma": "all", "singleQuote": false, - "printWidth": 120, + "printWidth": 100, "tabWidth": 4 } diff --git a/gmail/appsscript.json b/gmail/appsscript.json index 0f1164ef0..ae620ba33 100644 --- a/gmail/appsscript.json +++ b/gmail/appsscript.json @@ -22,33 +22,102 @@ "https://*.odoo.com/mail_plugin/get_translations", "https://*.odoo.com/mail_plugin/partner/get", "https://*.odoo.com/mail_plugin/log_mail_content", - "https://*.odoo.com/mail_plugin/partner/search", + "https://*.odoo.com/mail_plugin/search_records/res.partner", + "https://*.odoo.com/mail_plugin/redirect_to_record/res.partner", "https://*.odoo.com/mail_plugin/partner/create", "https://*.odoo.com/mail_plugin/partner/enrich_and_create_company", "https://*.odoo.com/mail_plugin/partner/enrich_and_update_company", + "https://*.odoo.com/mail_plugin/search_records/crm.lead", + "https://*.odoo.com/mail_plugin/redirect_to_record/crm.lead", "https://*.odoo.com/mail_plugin/lead/create", + "https://*.odoo.com/mail_plugin/search_records/helpdesk.ticket", + "https://*.odoo.com/mail_plugin/redirect_to_record/helpdesk.ticket", "https://*.odoo.com/mail_plugin/ticket/create", - "https://*.odoo.com/mail_plugin/project/search", + "https://*.odoo.com/mail_plugin/search_records/project.task", + "https://*.odoo.com/mail_plugin/redirect_to_record/project.task", + "https://*.odoo.com/mail_plugin/search_records/project.project", + "https://*.odoo.com/mail_plugin/redirect_to_record/project.project", "https://*.odoo.com/mail_plugin/project/create", "https://*.odoo.com/mail_plugin/task/create", "https://*.odoo.com/web/login", "https://*.odoo.com/mail_plugin/auth", "https://*.odoo.com/mail_plugin/auth/access_token", + "https://*.odoo.com/mail_plugin/auth/check_version", + "https://odoo.com/mail_plugin/get_translations", "https://odoo.com/mail_plugin/partner/get", "https://odoo.com/mail_plugin/log_mail_content", - "https://odoo.com/mail_plugin/partner/search", + "https://odoo.com/mail_plugin/search_records/res.partner", + "https://odoo.com/mail_plugin/redirect_to_record/res.partner", "https://odoo.com/mail_plugin/partner/create", "https://odoo.com/mail_plugin/partner/enrich_and_create_company", "https://odoo.com/mail_plugin/partner/enrich_and_update_company", + "https://odoo.com/mail_plugin/search_records/crm.lead", + "https://odoo.com/mail_plugin/redirect_to_record/crm.lead", + "https://odoo.com/mail_plugin/search_records/helpdesk.ticket", + "https://odoo.com/mail_plugin/redirect_to_record/helpdesk.ticket", "https://odoo.com/mail_plugin/lead/create", "https://odoo.com/mail_plugin/ticket/create", - "https://odoo.com/mail_plugin/project/search", + "https://odoo.com/mail_plugin/search_records/project.task", + "https://odoo.com/mail_plugin/redirect_to_record/project.task", + "https://odoo.com/mail_plugin/search_records/project.project", + "https://odoo.com/mail_plugin/redirect_to_record/project.project", "https://odoo.com/mail_plugin/project/create", "https://odoo.com/mail_plugin/task/create", "https://odoo.com/web/login", "https://odoo.com/mail_plugin/auth", "https://odoo.com/mail_plugin/auth/access_token", - "https://iap-services.odoo.com/iap/mail_extension/enrich" - ] + "https://odoo.com/mail_plugin/auth/check_version", + + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/get_translations", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/partner/get", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/log_mail_content", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/search_records/res.partner", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/redirect_to_record/res.partner", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/partner/create", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/partner/enrich_and_create_company", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/search_records/crm.lead", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/redirect_to_record/crm.lead", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/partner/enrich_and_update_company", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/search_records/helpdesk.ticket", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/redirect_to_record/helpdesk.ticket", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/lead/create", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/ticket/create", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/search_records/project.task", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/redirect_to_record/project.task", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/search_records/project.project", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/redirect_to_record/project.project", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/project/create", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/task/create", + "https://93835281-master-all.runbot154.odoo.com/web/login", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/auth", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/auth/access_token", + "https://93835281-master-all.runbot154.odoo.com/mail_plugin/auth/check_version", + + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/get_translations", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/partner/get", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/log_mail_content", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/search_records/res.partner", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/redirect_to_record/res.partner", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/partner/create", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/partner/enrich_and_create_company", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/search_records/crm.lead", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/redirect_to_record/crm.lead", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/partner/enrich_and_update_company", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/search_records/helpdesk.ticket", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/redirect_to_record/helpdesk.ticket", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/lead/create", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/ticket/create", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/search_records/project.task", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/redirect_to_record/project.task", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/search_records/project.project", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/redirect_to_record/project.project", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/project/create", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/task/create", + "https://93639085-18-0-all.runbot143.odoo.com/web/login", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/auth", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/auth/access_token", + "https://93639085-18-0-all.runbot143.odoo.com/mail_plugin/auth/check_version" + ], + "runtimeVersion": "V8" } diff --git a/gmail/package-lock.json b/gmail/package-lock.json index 93b2c28c3..b1429fc69 100644 --- a/gmail/package-lock.json +++ b/gmail/package-lock.json @@ -1,17 +1,279 @@ { + "name": "gmail", + "lockfileVersion": 3, "requires": true, - "lockfileVersion": 1, - "dependencies": { - "@types/google-apps-script": { - "version": "1.0.31", - "resolved": "https://registry.npmjs.org/@types/google-apps-script/-/google-apps-script-1.0.31.tgz", - "integrity": "sha512-tgsJKk20fwFoh0Ml4Li3pqKQ5uu3Nr3XeRsee2+pkPGrJxDlA3qsHAA2q3/HRv5yi9U6QVvdGwJ16USnmA7wAA==" - }, - "prettier": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.2.1.tgz", - "integrity": "sha512-PqyhM2yCjg/oKkFPtTGUojv7gnZAoG80ttl45O6x2Ug/rMJw4wcc9k6aaf2hibP7BGVCCM33gZoGjyvt9mm16Q==", + "packages": { + "": { + "devDependencies": { + "@rollup/plugin-node-resolve": "^15.0.2", + "@rollup/plugin-typescript": "^11.1.1", + "@types/google-apps-script": "^2.0.4", + "prettier": "^2.2.1", + "rollup": "^3.22.0", + "tslib": "^2.5.3" + } + }, + "node_modules/@rollup/plugin-node-resolve": { + "version": "15.3.1", + "resolved": "https://registry.npmjs.org/@rollup/plugin-node-resolve/-/plugin-node-resolve-15.3.1.tgz", + "integrity": "sha512-tgg6b91pAybXHJQMAAwW9VuWBO6Thi+q7BCNARLwSqlmsHz0XYURtGvh/AuwSADXSI4h/2uHbs7s4FzlZDGSGA==", + "dev": true, + "dependencies": { + "@rollup/pluginutils": "^5.0.1", + "@types/resolve": "1.20.2", + "deepmerge": "^4.2.2", + "is-module": "^1.0.0", + "resolve": "^1.22.1" + }, + "engines": { + "node": ">=14.0.0" + }, + "peerDependencies": { + "rollup": "^2.78.0||^3.0.0||^4.0.0" + }, + "peerDependenciesMeta": { + "rollup": { + "optional": true + } + } + }, + "node_modules/@rollup/plugin-typescript": { + "version": "11.1.6", + "resolved": "https://registry.npmjs.org/@rollup/plugin-typescript/-/plugin-typescript-11.1.6.tgz", + "integrity": "sha512-R92yOmIACgYdJ7dJ97p4K69I8gg6IEHt8M7dUBxN3W6nrO8uUxX5ixl0yU/N3aZTi8WhPuICvOHXQvF6FaykAA==", + "dev": true, + "dependencies": { + "@rollup/pluginutils": "^5.1.0", + "resolve": "^1.22.1" + }, + "engines": { + "node": ">=14.0.0" + }, + "peerDependencies": { + "rollup": "^2.14.0||^3.0.0||^4.0.0", + "tslib": "*", + "typescript": ">=3.7.0" + }, + "peerDependenciesMeta": { + "rollup": { + "optional": true + }, + "tslib": { + "optional": true + } + } + }, + "node_modules/@rollup/pluginutils": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/@rollup/pluginutils/-/pluginutils-5.3.0.tgz", + "integrity": "sha512-5EdhGZtnu3V88ces7s53hhfK5KSASnJZv8Lulpc04cWO3REESroJXg73DFsOmgbU2BhwV0E20bu2IDZb3VKW4Q==", + "dev": true, + "dependencies": { + "@types/estree": "^1.0.0", + "estree-walker": "^2.0.2", + "picomatch": "^4.0.2" + }, + "engines": { + "node": ">=14.0.0" + }, + "peerDependencies": { + "rollup": "^1.20.0||^2.0.0||^3.0.0||^4.0.0" + }, + "peerDependenciesMeta": { + "rollup": { + "optional": true + } + } + }, + "node_modules/@types/estree": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", + "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", + "dev": true + }, + "node_modules/@types/google-apps-script": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/@types/google-apps-script/-/google-apps-script-2.0.4.tgz", + "integrity": "sha512-Ra6vduULsar+WU2cn1h2TQoEXh45ty8jjjrdPqO4f/S8lbR75mYDI+5zjhZF7u2y2X3oB1qPqkSQBy7T7vdsnA==", + "dev": true + }, + "node_modules/@types/resolve": { + "version": "1.20.2", + "resolved": "https://registry.npmjs.org/@types/resolve/-/resolve-1.20.2.tgz", + "integrity": "sha512-60BCwRFOZCQhDncwQdxxeOEEkbc5dIMccYLwbxsS4TUNeVECQ/pBJ0j09mrHOl/JJvpRPGwO9SvE4nR2Nb/a4Q==", + "dev": true + }, + "node_modules/deepmerge": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.3.1.tgz", + "integrity": "sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/estree-walker": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", + "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==", "dev": true + }, + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/function-bind": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "dev": true, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/hasown": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "dev": true, + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/is-core-module": { + "version": "2.16.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", + "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", + "dev": true, + "dependencies": { + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-module": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-module/-/is-module-1.0.0.tgz", + "integrity": "sha512-51ypPSPCoTEIN9dy5Oy+h4pShgJmPCygKfyRCISBI+JoWT/2oJvK8QPxmwv7b/p239jXrm9M1mlQbyKJ5A152g==", + "dev": true + }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true + }, + "node_modules/picomatch": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz", + "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==", + "dev": true, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/prettier": { + "version": "2.8.8", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.8.8.tgz", + "integrity": "sha512-tdN8qQGvNjw4CHbY+XXk0JgCXn9QiF21a55rBe5LJAU+kDyC4WQn4+awm2Xfk2lQMk5fKup9XgzTZtGkjBdP9Q==", + "dev": true, + "bin": { + "prettier": "bin-prettier.js" + }, + "engines": { + "node": ">=10.13.0" + }, + "funding": { + "url": "https://github.com/prettier/prettier?sponsor=1" + } + }, + "node_modules/resolve": { + "version": "1.22.10", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.10.tgz", + "integrity": "sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w==", + "dev": true, + "dependencies": { + "is-core-module": "^2.16.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/rollup": { + "version": "3.29.5", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-3.29.5.tgz", + "integrity": "sha512-GVsDdsbJzzy4S/v3dqWPJ7EfvZJfCHiDqe80IyrF59LYuP+e6U1LJoUqeuqRbwAWoMNoXivMNeNAOf5E22VA1w==", + "dev": true, + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=14.18.0", + "npm": ">=8.0.0" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/tslib": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "dev": true + }, + "node_modules/typescript": { + "version": "5.9.2", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.2.tgz", + "integrity": "sha512-CWBzXQrc/qOkhidw1OzBTQuYRbfyxDXJMVJ1XNwUHGROVmuaeiEm3OslpZ1RV96d7SKKjZKrSJu3+t/xlw3R9A==", + "dev": true, + "peer": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } } } } diff --git a/gmail/package.json b/gmail/package.json index abe3215ab..7dfefad83 100644 --- a/gmail/package.json +++ b/gmail/package.json @@ -2,7 +2,7 @@ "devDependencies": { "@rollup/plugin-node-resolve": "^15.0.2", "@rollup/plugin-typescript": "^11.1.1", - "@types/google-apps-script": "^1.0.64", + "@types/google-apps-script": "^2.0.4", "prettier": "^2.2.1", "rollup": "^3.22.0", "tslib": "^2.5.3" diff --git a/gmail/src/const.ts b/gmail/src/const.ts index a59c7976e..091b039c6 100644 --- a/gmail/src/const.ts +++ b/gmail/src/const.ts @@ -1,28 +1,26 @@ export const URLS: Record = { GET_TRANSLATIONS: "/mail_plugin/get_translations", LOG_EMAIL: "/mail_plugin/log_mail_content", + SEARCH_RECORDS: "/mail_plugin/search_records", // Partner GET_PARTNER: "/mail_plugin/partner/get", - SEARCH_PARTNER: "/mail_plugin/partner/search", + SEARCH_PARTNER: "/mail_plugin/search_records/res.partner", PARTNER_CREATE: "/mail_plugin/partner/create", - CREATE_COMPANY: "/mail_plugin/partner/enrich_and_create_company", - ENRICH_COMPANY: "/mail_plugin/partner/enrich_and_update_company", // CRM Lead CREATE_LEAD: "/mail_plugin/lead/create", // HELPDESK Ticket CREATE_TICKET: "/mail_plugin/ticket/create", // Project - SEARCH_PROJECT: "/mail_plugin/project/search", + SEARCH_PROJECT: "/mail_plugin/search_records/project.project", CREATE_PROJECT: "/mail_plugin/project/create", CREATE_TASK: "/mail_plugin/task/create", - // IAP - IAP_COMPANY_ENRICHMENT: "https://iap-services.odoo.com/iap/mail_extension/enrich", }; export const ODOO_AUTH_URLS: Record = { LOGIN: "/web/login", AUTH_CODE: "/mail_plugin/auth", CODE_VALIDATION: "/mail_plugin/auth/access_token", + CHECK_VERSION: "/mail_plugin/auth/check_version", SCOPE: "outlook", FRIENDLY_NAME: "Gmail", }; diff --git a/gmail/src/main.ts b/gmail/src/main.ts index 028699406..7751c0e14 100644 --- a/gmail/src/main.ts +++ b/gmail/src/main.ts @@ -3,6 +3,7 @@ import { Email } from "./models/email"; import { State } from "./models/state"; import { Partner } from "./models/partner"; import { _t } from "./services/translation"; +import { buildLoginMainView } from "./views/login"; /** * Entry point of the application, executed when an email is open. @@ -17,26 +18,36 @@ function onGmailMessageOpen(event) { GmailApp.setCurrentMessageAccessToken(event.messageMetadata.accessToken); const currentEmail = new Email(event.gmail.messageId, event.gmail.accessToken); - const [partner, odooUserCompanies, canCreatePartner, canCreateProject, error] = Partner.enrichPartner( - currentEmail.contactEmail, - currentEmail.contactName - ); + let state = null; + if (currentEmail.contacts.length > 1) { + // More than one contact, we will need to choose the right one + const [searchedPartners, error] = Partner.searchPartner( + currentEmail.contacts.map((c) => c.email), + ); + if (error.code) { + return buildLoginMainView(); + } + const existingPartnersEmails = searchedPartners.map((p) => p.email); - if (!partner) { - // Should at least use the FROM headers to generate the partner - throw new Error(_t("Error during enrichment")); - } + for (const contact of currentEmail.contacts) { + if (existingPartnersEmails.includes(contact.email)) { + continue; + } + searchedPartners.push(Partner.fromJson({ name: contact.name, email: contact.email })); + } + + state = new State(null, false, currentEmail, searchedPartners, null, false); + } else { + const [partner, canCreatePartner, canCreateProject, error] = Partner.getPartner( + currentEmail.contacts[0].name, + currentEmail.contacts[0].email, + ); + if (error.code) { + return buildLoginMainView(); + } - const state = new State( - partner, - canCreatePartner, - currentEmail, - odooUserCompanies, - null, - null, - canCreateProject, - error - ); + state = new State(partner, canCreatePartner, currentEmail, null, null, canCreateProject); + } return [buildView(state)]; } diff --git a/gmail/src/models/company.ts b/gmail/src/models/company.ts deleted file mode 100644 index ab0483ccd..000000000 --- a/gmail/src/models/company.ts +++ /dev/null @@ -1,152 +0,0 @@ -import { formatUrl, isTrue, first } from "../utils/format"; - -export class Company { - id: number; - name: string; - email: string; - phone: string; - isEnriched: boolean; - - // Additional Information - address: string; - annualRevenue: string; - companyType: string; - description: string; - emails: string; - employees: number; - foundedYear: number; - image: string; - industry: string; - mobile: string; - phones: string; - tags: string; - timezone: string; - timezoneUrl: string; - twitterFollowers: number; - twitterBio: string; - website: string; - - // Social Medias - crunchbase: string; - facebook: string; - linkedin: string; - twitter: string; - - /** - * Parse the dictionary returned by IAP. - */ - static fromIapResponse(values: any): Company { - const company = new Company(); - - company.name = isTrue(values.name); - company.email = first(values.email); - company.phone = first(values.phone_numbers); - company.isEnriched = !!Object.keys(values).length; - - company.emails = isTrue(values.email) ? values.email.join("\n") : null; - company.phones = isTrue(values.phone_numbers) ? values.phone_numbers.join("\n") : null; - - company.image = isTrue(values.logo); - company.website = formatUrl(values.domain); - company.description = isTrue(values.description); - company.address = isTrue(values.location); - - // Social Medias - company.facebook = isTrue(values.facebook); - company.twitter = isTrue(values.twitter); - company.linkedin = isTrue(values.linkedin); - company.crunchbase = isTrue(values.crunchbase); - - // Additional Information - company.employees = values.employees || null; - company.annualRevenue = isTrue(values.estimated_annual_revenue); - company.industry = isTrue(values.industry); - company.twitterBio = isTrue(values.twitter_bio); - company.twitterFollowers = values.twitter_followers || null; - company.foundedYear = values.founded_year; - company.timezone = isTrue(values.timezone); - company.timezoneUrl = isTrue(values.timezone_url); - company.tags = isTrue(values.tag) ? values.tag.join(", ") : null; - company.companyType = isTrue(values.company_type); - - return company; - } - - /** - * Unserialize the company object (reverse JSON.stringify). - */ - static fromJson(values: any): Company { - const company = new Company(); - - company.id = values.id; - company.name = values.name; - company.email = values.email; - company.phone = values.phone; - - company.address = values.address; - company.annualRevenue = values.annualRevenue; - company.companyType = values.companyType; - company.description = values.description; - company.emails = values.emails; - company.employees = values.employees; - company.foundedYear = values.foundedYear; - company.image = values.image; - company.industry = values.industry; - company.mobile = values.mobile; - company.phones = values.phones; - company.tags = values.tags; - company.timezone = values.timezone; - company.timezoneUrl = values.timezoneUrl; - company.twitterBio = values.twitterBio; - company.twitterFollowers = values.twitterFollowers; - company.website = values.website; - - company.crunchbase = values.crunchbase; - company.facebook = values.facebook; - company.twitter = values.twitter; - company.linkedin = values.linkedin; - - return company; - } - - /** - * Parse the dictionary returned by an Odoo database. - */ - static fromOdooResponse(values: any): Company { - if (!values.id || values.id < 0) { - return null; - } - - const iapInfo = values.additionalInfo || {}; - - const company = this.fromIapResponse(iapInfo); - - // Overwrite IAP information with the Odoo client database information - company.id = values.id; - company.name = values.name; - company.email = values.email; - company.phone = values.phone; - - company.mobile = values.mobile; - company.website = values.website; - company.image = values.image ? "data:image/png;base64," + values.image : null; - - if (values.address) { - company.address = ""; - - if (isTrue(values.address.street)) { - company.address += values.address.street + ", "; - } - if (isTrue(values.address.zip)) { - company.address += values.address.zip + " "; - } - if (isTrue(values.address.city)) { - company.address += values.address.city + " "; - } - if (isTrue(values.address.country)) { - company.address += values.address.country; - } - } - return company; - } -} diff --git a/gmail/src/models/email.ts b/gmail/src/models/email.ts index f623e8e70..6161e6a4e 100644 --- a/gmail/src/models/email.ts +++ b/gmail/src/models/email.ts @@ -7,10 +7,16 @@ export class Email { accessToken: string; messageId: string; subject: string; + body: string; + timestamp: number; - contactEmail: string; - contactFullEmail: string; - contactName: string; + emailFrom: string; + contacts: EmailContact[]; + + // When asking for the attachments, a long moment after opening + // the addon, then the token to get the Gmail Message expired + // so we cache the result and ask it when loading the app + _attachmentsParsed: [string[][], ErrorMessage]; constructor(messageId: string = null, accessToken: string = null) { if (messageId) { @@ -21,51 +27,68 @@ export class Email { this.messageId = messageId; const message = GmailApp.getMessageById(this.messageId); this.subject = message.getSubject(); - - const fromHeaders = message.getFrom(); - const sent = fromHeaders.toLowerCase().indexOf(userEmail) >= 0; - this.contactFullEmail = sent ? message.getTo() : message.getFrom(); - [this.contactName, this.contactEmail] = this._emailSplitTuple(this.contactFullEmail); + this.body = message.getBody(); + this.timestamp = message.getDate().getTime(); + this.emailFrom = message.getFrom(); + + this._attachmentsParsed = this.getAttachments(); + + this.contacts = [ + ...this._emailSplitTuple(message.getTo(), userEmail), + ...this._emailSplitTuple(this.emailFrom, userEmail), + ...this._emailSplitTuple(message.getCc(), userEmail), + ...this._emailSplitTuple(message.getBcc(), userEmail), + ]; } } /** - * Ask the email body only if the user asked for it (e.g. asked to log the email). - */ - public get body() { - GmailApp.setCurrentMessageAccessToken(this.accessToken); - const message = GmailApp.getMessageById(this.messageId); - return message.getBody(); - } - - /** - * Parse a full FROM header and return the name part and the email part. + * Parse a full FROM header and return the name and email parts. * * E.G. - * "BOB" => ["BOB", "bob@example.com"] - * bob@example.com => ["bob@example.com", "bob@example.com"] + * "BOB" + * => [["BOB", "bob@example.com"]] * + * bob@example.com + * => [["bob@example.com", "bob@example.com"]] + * + * alice@example.com, bob@example.com + * => [ + * ["alice@example.com", "alice@example.com"], + * ["bob@example.com", "bob@example.com"] + * ] + * + * "Alice" , "BOB" + * => [ + * ["alice@example.com", "alice@example.com"], + * ["bob@example.com", "bob@example.com"] + * ] + * + * , + * => [ + * ["alice@example.com", "alice@example.com"], + * ["bob@example.com", "bob@example.com"] + * ] */ - _emailSplitTuple(fullEmail: string): [string, string] { - const match = fullEmail.match(/(.*)<(.*)>/); - fullEmail = fullEmail.replace("<", "").replace(">", ""); - - if (!match) { - return [fullEmail, fullEmail]; - } - - const [_, name, email] = match; - - if (!name || !email) { - return [fullEmail, fullEmail]; - } + _emailSplitTuple(fullEmail: string, userEmail: string): EmailContact[] { + const contacts = []; + const re = /(.*?)<(.*?)>/; + for (const part of fullEmail.split(",")) { + if (part.toLowerCase().indexOf(userEmail) >= 0 || !part.trim()?.length) { + // Skip the user's email + continue; + } - const cleanedName = name.replace(/\"/g, "").trim(); - if (!cleanedName || !cleanedName.length) { - return [fullEmail, fullEmail]; + const result = part.match(re); + if (!result) { + contacts.push(new EmailContact(part.trim(), part.trim(), part.trim())); + continue; + } + const email = result[2].trim(); + let name = result[1].replace(/\"/g, "").trim() || email; + contacts.push(new EmailContact(name, email, part.trim())); } - - return [cleanedName, email]; + return contacts; } /** @@ -73,15 +96,17 @@ export class Email { */ static fromJson(values: any): Email { const email = new Email(); - email.accessToken = values.accessToken; email.messageId = values.messageId; email.subject = values.subject; - - email.contactEmail = values.contactEmail; - email.contactFullEmail = values.contactFullEmail; - email.contactName = values.contactName; - + email.body = values.body; + email.timestamp = values.timestamp; + email.emailFrom = values.emailFrom; + email.contacts = values.contacts.map((c) => EmailContact.fromJson(c)); + email._attachmentsParsed = [ + values._attachmentsParsed[0], + ErrorMessage.fromJson(values._attachmentsParsed[1]), + ]; return email; } @@ -97,6 +122,9 @@ export class Email { * - Otherwise, the list of attachments base 64 encoded and an empty error message */ getAttachments(): [string[][], ErrorMessage] { + if (this._attachmentsParsed) { + return this._attachmentsParsed; + } GmailApp.setCurrentMessageAccessToken(this.accessToken); const message = GmailApp.getMessageById(this.messageId); const gmailAttachments = message.getAttachments(); @@ -124,3 +152,19 @@ export class Email { return [attachments, new ErrorMessage(null)]; } } + +export class EmailContact { + name: string; + email: string; + fullEmail: string; + + constructor(name: string, email: string, fullEmail: string) { + this.name = name; + this.email = email; + this.fullEmail = fullEmail; + } + + static fromJson(values: any): EmailContact { + return new EmailContact(values.name, values.email, values.fullEmail); + } +} diff --git a/gmail/src/models/error_message.ts b/gmail/src/models/error_message.ts index 05153ec7b..5bad47967 100644 --- a/gmail/src/models/error_message.ts +++ b/gmail/src/models/error_message.ts @@ -7,14 +7,6 @@ import { _t } from "../services/translation"; const _ERROR_CODE_MESSAGES: Record = { odoo: null, // Message is contained in the additional information http_error_odoo: "Could not connect to database. Try to log out and in.", - insufficient_credit: "Not enough credits to enrich.", - company_created: null, - company_updated: null, - // IAP - http_error_iap: "Our IAP server is down, please come back later.", - exhausted_requests: - "Oops, looks like you have exhausted your free enrichment requests. Please log in to try again.", - missing_data: "No insights found for this address", unknown: "Something bad happened. Please, try again later.", // Attachment attachments_size_exceeded: @@ -30,12 +22,6 @@ export class ErrorMessage { message: string; information: string; - // False if the error means that we can not contact the Odoo database - // (e.g. HTTP error) - canContactOdooDatabase: boolean = true; - - canCreateCompany: boolean = true; - constructor(code: string = null, information: any = null) { if (code) { this.setError(code, information); @@ -53,11 +39,7 @@ export class ErrorMessage { this.code = code; this.information = information; - this.message = _t(_ERROR_CODE_MESSAGES[this.code]); - - if (code === "http_error_odoo") { - this.canContactOdooDatabase = false; - } + this.message = information || _t(_ERROR_CODE_MESSAGES[this.code]); } /** @@ -67,8 +49,6 @@ export class ErrorMessage { const error = new ErrorMessage(); error.code = values.code; error.message = values.message; - error.canContactOdooDatabase = values.canContactOdooDatabase; - error.canCreateCompany = values.canCreateCompany; error.information = values.information; return error; } diff --git a/gmail/src/models/lead.ts b/gmail/src/models/lead.ts index dce33c26c..6062a99ff 100644 --- a/gmail/src/models/lead.ts +++ b/gmail/src/models/lead.ts @@ -1,7 +1,9 @@ import { postJsonRpc } from "../utils/http"; -import { isTrue } from "../utils/format"; import { URLS } from "../const"; import { getAccessToken } from "src/services/odoo_auth"; +import { _t } from "../services/translation"; +import { Partner } from "./partner"; +import { Email } from "./email"; /** * Represent a "crm.lead" record. @@ -9,26 +11,39 @@ import { getAccessToken } from "src/services/odoo_auth"; export class Lead { id: number; name: string; - expectedRevenue: string; - probability: number; - recurringRevenue: string; - recurringPlan: string; + revenuesDescription: string; /** * Make a RPC call to the Odoo database to create a lead * and return the ID of the newly created record. */ - static createLead(partnerId: number, emailBody: string, emailSubject: string): number { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.CREATE_LEAD; + static createLead(partner: Partner, email: Email): [Lead, Partner] | null { + const url = + PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.CREATE_LEAD; const accessToken = getAccessToken(); - + const [attachments, _] = email.getAttachments(); const response = postJsonRpc( url, - { email_body: emailBody, email_subject: emailSubject, partner_id: partnerId }, + { + email_body: email.body, + email_subject: email.subject, + partner_id: partner.id, + partner_email: partner.email, + partner_name: partner.name, + attachments, + }, { Authorization: "Bearer " + accessToken }, ); - return response ? response.lead_id || null : null; + if (!response?.id) { + return null; + } + if (!partner.id) { + partner.id = response.partner_id; + partner.image = response.partner_image; + partner.isWritable = true; + } + return [Lead.fromOdooResponse(response), partner]; } /** @@ -38,10 +53,7 @@ export class Lead { const lead = new Lead(); lead.id = values.id; lead.name = values.name; - lead.expectedRevenue = values.expectedRevenue; - lead.probability = values.probability; - lead.recurringRevenue = values.recurringRevenue; - lead.recurringPlan = values.recurringPlan; + lead.revenuesDescription = values.revenuesDescription; return lead; } @@ -50,16 +62,9 @@ export class Lead { */ static fromOdooResponse(values: any): Lead { const lead = new Lead(); - lead.id = values.lead_id; + lead.id = values.id; lead.name = values.name; - lead.expectedRevenue = values.expected_revenue; - lead.probability = values.probability; - - if (isTrue(values.recurring_revenue) && isTrue(values.recurring_plan)) { - lead.recurringRevenue = values.recurring_revenue; - lead.recurringPlan = values.recurring_plan; - } - + lead.revenuesDescription = values.revenues_description; return lead; } } diff --git a/gmail/src/models/partner.ts b/gmail/src/models/partner.ts index 0ac450ea7..17c626a03 100644 --- a/gmail/src/models/partner.ts +++ b/gmail/src/models/partner.ts @@ -1,12 +1,12 @@ -import { Company } from "./company"; import { Lead } from "./lead"; import { Task } from "./task"; import { Ticket } from "./ticket"; -import { postJsonRpc, postJsonRpcCached } from "../utils/http"; +import { postJsonRpc } from "../utils/http"; import { URLS } from "../const"; import { ErrorMessage } from "../models/error_message"; import { getAccessToken } from "src/services/odoo_auth"; import { getOdooServerUrl } from "src/services/app_properties"; +import { UI_ICONS } from "../views/icons"; /** * Represent the current partner and all the information about him. @@ -18,15 +18,28 @@ export class Partner { image: string; isCompany: boolean; + companyName: string; phone: string; mobile: string; - company: Company; leads: Lead[]; + leadCount: number; tickets: Ticket[]; + ticketCount: number; tasks: Task[]; + taskCount: number; - isWriteable: boolean; + isWritable: boolean; + + /** + * Return the image to show in the interface for the current partner. + */ + getImage() { + if (!this.id || this.id < 0 || !this.image) { + return UI_ICONS.person; + } + return this.image; + } /** * Unserialize the partner object (reverse JSON.stringify). @@ -40,19 +53,27 @@ export class Partner { partner.image = values.image; partner.isCompany = values.isCompany; + partner.companyName = values.companyName; partner.phone = values.phone; partner.mobile = values.mobile; - partner.company = values.company ? Company.fromJson(values.company) : null; - partner.isWriteable = values.isWriteable; + partner.leadCount = values.leadCount; + partner.ticketCount = values.ticketCount; + partner.taskCount = values.taskCount; - partner.leads = values.leads ? values.leads.map((leadValues: any) => Lead.fromJson(leadValues)) : null; + partner.isWritable = values.isWritable; + + partner.leads = values.leads + ? values.leads.map((leadValues: any) => Lead.fromJson(leadValues)) + : null; partner.tickets = values.tickets ? values.tickets.map((ticketValues: any) => Ticket.fromJson(ticketValues)) : null; - partner.tasks = values.tasks ? values.tasks.map((taskValues: any) => Task.fromJson(taskValues)) : null; + partner.tasks = values.tasks + ? values.tasks.map((taskValues: any) => Task.fromJson(taskValues)) + : null; return partner; } @@ -66,86 +87,43 @@ export class Partner { partner.name = values.name; partner.email = values.email; - partner.image = values.image ? "data:image/png;base64," + values.image : null; + partner.image = values.image; + partner.isCompany = values.is_company; partner.isCompany = values.is_company; + partner.companyName = values.company_name; + partner.phone = values.phone; partner.mobile = values.mobile; - - // Undefined should be considered as True for retro-compatibility - partner.isWriteable = values.can_write_on_partner !== false; - - if (values.company && values.company.id && values.company.id > 0) { - partner.company = Company.fromOdooResponse(values.company); - } + partner.isWritable = values.can_write_on_partner; return partner; } - /** - * Try to find information about the given email /name. - * - * If we are not logged to an Odoo database, enrich the email domain with IAP. - * Otherwise fetch the partner on the user database. - * - * See `getPartner` - */ - static enrichPartner(email: string, name: string): [Partner, number[], boolean, boolean, ErrorMessage] { - const odooServerUrl = getOdooServerUrl(); - const odooAccessToken = getAccessToken(); - - if (odooServerUrl && odooAccessToken) { - return this.getPartner(email, name); - } else { - const [partner, error] = this._enrichFromIap(email, name); - return [partner, null, false, false, error]; - } - } - - /** - * Extract the email domain and send a request to IAP - * to find information about the company. - */ - static _enrichFromIap(email: string, name: string): [Partner, ErrorMessage] { - const odooSharedSecret = PropertiesService.getScriptProperties().getProperty("ODOO_SHARED_SECRET"); - const userEmail = Session.getEffectiveUser().getEmail(); - - const senderDomain = email.split("@").pop(); - - const response = postJsonRpcCached(URLS.IAP_COMPANY_ENRICHMENT, { - email: userEmail, - domain: senderDomain, - secret: odooSharedSecret, - }); - - const error = new ErrorMessage(); - if (!response) { - error.setError("http_error_iap"); - } else if (response.error && response.error.length) { - error.setError(response.error); - } - - const partner = new Partner(); - partner.name = name; - partner.email = email; - - if (response && response.name) { - partner.company = Company.fromIapResponse(response); - } - - return [partner, error]; - } /** * Create a "res.partner" with the given values in the Odoo database. */ - static savePartner(partnerValues: any): number { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.PARTNER_CREATE; + static savePartner(partner: Partner): Partner | null { + const url = + PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + + URLS.PARTNER_CREATE; const odooAccessToken = getAccessToken(); + const partnerValues = { + name: partner.name, + email: partner.email, + }; + const response = postJsonRpc(url, partnerValues, { Authorization: "Bearer " + odooAccessToken, }); - return response && response.id; + if (!response?.id) { + return null; + } + partner.id = response.id; + partner.image = response.image; + partner.isWritable = true; + return partner; } /** @@ -153,119 +131,101 @@ export class Partner { * * Return * - The Partner related to the given email address - * - The list of Odoo companies in which the current user belongs * - True if the current user can create partner in his Odoo database * - True if the current user can create projects in his Odoo database * - The error message if something bad happened */ static getPartner( - email: string, name: string, + email: string, partnerId: number = null, - ): [Partner, number[], boolean, boolean, ErrorMessage] { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.GET_PARTNER; + ): [Partner, boolean, boolean, ErrorMessage] { + const odooServerUrl = getOdooServerUrl(); const odooAccessToken = getAccessToken(); + if (!odooServerUrl || !odooAccessToken) { + const error = new ErrorMessage("http_error_odoo"); + const partner = Partner.fromJson({ name, email }); + return [partner, false, false, error]; + } + + const url = + PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.GET_PARTNER; + const response = postJsonRpc( url, - { email: email, name: name, partner_id: partnerId }, + { email: email, partner_id: partnerId }, { Authorization: "Bearer " + odooAccessToken }, ); + if (response && response.error) { + const error = new ErrorMessage("odoo", response.error); + const partner = Partner.fromJson({ name, email }); + return [partner, false, false, error]; + } + if (!response || !response.partner) { const error = new ErrorMessage("http_error_odoo"); - const partner = Partner.fromJson({ name: name, email: email }); - return [partner, null, false, false, error]; + const partner = Partner.fromJson({ name, email }); + return [partner, false, false, error]; } const error = new ErrorMessage(); - - if (response.enrichment_info && response.enrichment_info.type) { - error.setError(response.enrichment_info.type, response.enrichment_info.info); - } else if (response.partner.enrichment_info && response.partner.enrichment_info.type) { - error.setError(response.partner.enrichment_info.type, response.partner.enrichment_info.info); - } - - const partner = Partner.fromOdooResponse(response.partner); + const partner = Partner.fromOdooResponse({ name, email, ...response.partner }); // Parse leads if (response.leads) { - partner.leads = response.leads.map((leadValues: any) => Lead.fromOdooResponse(leadValues)); + partner.leadCount = response.lead_count; + partner.leads = response.leads.map((leadValues: any) => + Lead.fromOdooResponse(leadValues), + ); } // Parse tickets if (response.tickets) { - partner.tickets = response.tickets.map((ticketValues: any) => Ticket.fromOdooResponse(ticketValues)); + partner.ticketCount = response.ticket_count; + partner.tickets = response.tickets.map((ticketValues: any) => + Ticket.fromOdooResponse(ticketValues), + ); } // Parse tasks if (response.tasks) { - partner.tasks = response.tasks.map((taskValues: any) => Task.fromOdooResponse(taskValues)); + partner.taskCount = response.task_count; + partner.tasks = response.tasks.map((taskValues: any) => + Task.fromOdooResponse(taskValues), + ); } const canCreateProject = response.can_create_project !== false; - const odooUserCompanies = response.user_companies || null; // undefined must be considered as true const canCreatePartner = response.can_create_partner !== false; - return [partner, odooUserCompanies, canCreatePartner, canCreateProject, error]; + return [partner, canCreatePartner, canCreateProject, error]; } /** * Perform a search on the Odoo database and return the list of matched partners. */ - static searchPartner(query: string): [Partner[], ErrorMessage] { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.SEARCH_PARTNER; + static searchPartner(query: string | string[]): [Partner[], ErrorMessage] { + const url = + PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + + URLS.SEARCH_PARTNER; const odooAccessToken = getAccessToken(); - const response = postJsonRpc(url, { search_term: query }, { Authorization: "Bearer " + odooAccessToken }); + const response = postJsonRpc( + url, + { query }, + { Authorization: "Bearer " + odooAccessToken }, + ); - if (!response || !response.partners) { + if (!response?.length) { return [[], new ErrorMessage("http_error_odoo")]; } - return [response.partners.map((values: any) => Partner.fromOdooResponse(values)), new ErrorMessage()]; - } - - /** - * Create and enrich the company of the given partner. - */ - static createCompany(partnerId: number): [Company, ErrorMessage] { - return this._enrichOrCreateCompany(partnerId, URLS.CREATE_COMPANY); - } - - /** - * Enrich the existing company. - */ - static enrichCompany(companyId: number): [Company, ErrorMessage] { - return this._enrichOrCreateCompany(companyId, URLS.ENRICH_COMPANY); - } - - static _enrichOrCreateCompany(partnerId: number, endpoint: string): [Company, ErrorMessage] { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + endpoint; - const odooAccessToken = getAccessToken(); - - const response = postJsonRpc(url, { partner_id: partnerId }, { Authorization: "Bearer " + odooAccessToken }); - - if (!response) { - return [null, new ErrorMessage("http_error_odoo")]; - } - - if (response.error) { - return [null, new ErrorMessage("odoo", response.error)]; - } - - let error = new ErrorMessage(); - - if (response.enrichment_info && response.enrichment_info.type) { - error.setError(response.enrichment_info.type, response.enrichment_info.info); - } - - if (error.code) { - error.canCreateCompany = false; - } - - const company = response.company ? Company.fromOdooResponse(response.company) : null; - return [company, error]; + return [ + response[0].map((values: any) => Partner.fromOdooResponse(values)), + new ErrorMessage(), + ]; } } diff --git a/gmail/src/models/project.ts b/gmail/src/models/project.ts index 02202e00d..c8ec03433 100644 --- a/gmail/src/models/project.ts +++ b/gmail/src/models/project.ts @@ -10,6 +10,8 @@ export class Project { id: number; name: string; partnerName: string; + stageName: string; + companyName: string; /** * Unserialize the project object (reverse JSON.stringify). @@ -19,6 +21,8 @@ export class Project { project.id = values.id; project.name = values.name; project.partnerName = values.partnerName; + project.stageName = values.stageName; + project.companyName = values.companyName; return project; } @@ -27,9 +31,11 @@ export class Project { */ static fromOdooResponse(values: any): Project { const project = new Project(); - project.id = values.project_id; + project.id = values.id; project.name = values.name; project.partnerName = values.partner_name; + project.stageName = values.stage_name; + project.companyName = values.company_name; return project; } @@ -37,16 +43,25 @@ export class Project { * Make a RPC call to the Odoo database to search a project. */ static searchProject(query: string): [Project[], ErrorMessage] { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.SEARCH_PROJECT; + const url = + PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + + URLS.SEARCH_PROJECT; const odooAccessToken = getAccessToken(); - const response = postJsonRpc(url, { search_term: query }, { Authorization: "Bearer " + odooAccessToken }); + const response = postJsonRpc( + url, + { query }, + { Authorization: "Bearer " + odooAccessToken }, + ); - if (!response) { + if (!response?.length) { return [[], new ErrorMessage("http_error_odoo")]; } - return [response.map((values: any) => Project.fromOdooResponse(values)), new ErrorMessage()]; + return [ + response[0].map((values: any) => Project.fromOdooResponse(values)), + new ErrorMessage(), + ]; } /** @@ -54,12 +69,18 @@ export class Project { * and return the newly created record. */ static createProject(name: string): Project { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.CREATE_PROJECT; + const url = + PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + + URLS.CREATE_PROJECT; const odooAccessToken = getAccessToken(); - const response = postJsonRpc(url, { name: name }, { Authorization: "Bearer " + odooAccessToken }); + const response = postJsonRpc( + url, + { name: name }, + { Authorization: "Bearer " + odooAccessToken }, + ); - const projectId = response ? response.project_id || null : null; + const projectId = response ? response.id || null : null; if (!projectId) { return null; } diff --git a/gmail/src/models/state.ts b/gmail/src/models/state.ts index 1760dbaf5..d48cc639b 100644 --- a/gmail/src/models/state.ts +++ b/gmail/src/models/state.ts @@ -1,8 +1,6 @@ -import { isTrue } from "../utils/format"; import { Email } from "./email"; import { Partner } from "./partner"; import { Project } from "./project"; -import { Lead } from "./lead"; import { ErrorMessage } from "./error_message"; import { getAccessToken, getOdooAuthUrl } from "../services/odoo_auth"; import { getOdooServerUrl } from "src/services/app_properties"; @@ -10,7 +8,7 @@ import { getOdooServerUrl } from "src/services/app_properties"; /** * Object which contains all data for the application. * - * In App-Script, all event handler are function and not method. We can only pass string + * In App-Script, all event handlers are function and not method. We can only pass string * as arguments. So this object is serialized, then given to the event handler and then * unserialize to retrieve the original object. * @@ -24,36 +22,26 @@ export class State { canCreatePartner: boolean; // Opened email with headers email: Email; - // ID list of the Odoo user companies - odooUserCompanies: number[]; // Searched partners in the search view searchedPartners: Partner[]; // Searched projects in the search view searchedProjects: Project[]; canCreateProject: boolean; - // Current error message displayed on the card - error: ErrorMessage; - // Used in the company card - isCompanyDescriptionUnfolded: boolean; constructor( partner: Partner, canCreatePartner: boolean, email: Email, - odooUserCompanies: number[], partners: Partner[], searchedProjects: Project[], canCreateProject: boolean, - error: ErrorMessage, ) { this.partner = partner; this.canCreatePartner = canCreatePartner; this.email = email; - this.odooUserCompanies = odooUserCompanies; this.searchedPartners = partners; this.searchedProjects = searchedProjects; this.canCreateProject = canCreateProject; - this.error = error; } toJson(): string { @@ -77,7 +65,6 @@ export class State { const partner = Partner.fromJson(partnerValues); const email = Email.fromJson(emailValues); const error = ErrorMessage.fromJson(errorValues); - const odooUserCompanies = values.odooUserCompanies; const searchedPartners = partnersValues ? partnersValues.map((partnerValues: any) => Partner.fromJson(partnerValues)) : null; @@ -85,36 +72,16 @@ export class State { ? projectsValues.map((projectValues: any) => Project.fromJson(projectValues)) : null; - // "isCompanyDescriptionUnfolded" is not copied - // to re-fold the description if we go back / refresh - return new State( partner, canCreatePartner, email, - odooUserCompanies, searchedPartners, searchedProjects, canCreateProject, - error, ); } - /** - * Return the companies of the Odoo user as a GET parameter to add in a URL or an - * empty string if the information is missing. - * - * e.g. - * &cids=1,3,7 - */ - get odooCompaniesParameter(): string { - if (this.odooUserCompanies && this.odooUserCompanies.length) { - const cids = this.odooUserCompanies.sort().join(","); - return `&cids=${cids}`; - } - return ""; - } - /** * Cache / user properties management. * @@ -123,7 +90,7 @@ export class State { */ static get accessToken() { const accessToken = getAccessToken(); - return isTrue(accessToken); + return accessToken?.length && accessToken; } static get isLogged(): boolean { @@ -135,16 +102,7 @@ export class State { */ static get odooLoginUrl(): string { const loginUrl = getOdooAuthUrl(); - return isTrue(loginUrl); - } - /** - * Return the shared secret between the add-on and IAP - * (which is used to authenticate the add-on to IAP). - */ - static get odooSharedSecret(): string { - const scriptProperties = PropertiesService.getScriptProperties(); - const sharedSecret = scriptProperties.getProperty("ODOO_SHARED_SECRET"); - return isTrue(sharedSecret); + return loginUrl?.length && loginUrl; } /** @@ -162,13 +120,15 @@ export class State { */ static getLoggingState(messageId: string) { const cache = CacheService.getUserCache(); - const loggingStateStr = cache.get("ODOO_LOGGING_STATE_" + getOdooServerUrl() + "_" + messageId); + const loggingStateStr = cache.get( + "ODOO_LOGGING_STATE_" + getOdooServerUrl() + "_" + messageId, + ); const defaultValues: Record = { - partners: [], - leads: [], - tickets: [], - tasks: [], + "res.partner": [], + "crm.lead": [], + "helpdesk.ticket": [], + "project.task": [], }; if (!loggingStateStr || !loggingStateStr.length) { diff --git a/gmail/src/models/task.ts b/gmail/src/models/task.ts index ba8b54530..3a124a36e 100644 --- a/gmail/src/models/task.ts +++ b/gmail/src/models/task.ts @@ -1,6 +1,8 @@ import { postJsonRpc } from "../utils/http"; import { URLS } from "../const"; import { getAccessToken } from "src/services/odoo_auth"; +import { Partner } from "./partner"; +import { Email } from "./email"; /** * Represent a "project.task" record. @@ -26,7 +28,7 @@ export class Task { */ static fromOdooResponse(values: any): Task { const task = new Task(); - task.id = values.task_id; + task.id = values.id; task.name = values.name; task.projectName = values.project_name; return task; @@ -36,25 +38,32 @@ export class Task { * Make a RPC call to the Odoo database to create a task * and return the ID of the newly created record. */ - static createTask(partnerId: number, projectId: number, emailBody: string, emailSubject: string): Task { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.CREATE_TASK; + static createTask(partner: Partner, projectId: number, email: Email): [Task, Partner] | null { + const url = + PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.CREATE_TASK; const odooAccessToken = getAccessToken(); - + const [attachments, _] = email.getAttachments(); const response = postJsonRpc( url, - { email_subject: emailSubject, email_body: emailBody, project_id: projectId, partner_id: partnerId }, + { + email_body: email.body, + email_subject: email.subject, + partner_email: partner.email, + partner_id: partner.id, + partner_name: partner.name, + project_id: projectId, + attachments, + }, { Authorization: "Bearer " + odooAccessToken }, ); - - const taskId = response ? response.task_id || null : null; - - if (!taskId) { + if (!response?.id) { return null; } - - return Task.fromJson({ - id: taskId, - name: response.name, - }); + if (!partner.id) { + partner.id = response.partner_id; + partner.image = response.partner_image; + partner.isWritable = true; + } + return [Task.fromOdooResponse(response), partner]; } } diff --git a/gmail/src/models/ticket.ts b/gmail/src/models/ticket.ts index fec7f420f..c311cc629 100644 --- a/gmail/src/models/ticket.ts +++ b/gmail/src/models/ticket.ts @@ -1,6 +1,8 @@ import { postJsonRpc } from "../utils/http"; import { URLS } from "../const"; import { getAccessToken } from "src/services/odoo_auth"; +import { Partner } from "./partner"; +import { Email } from "./email"; /** * Represent a "helpdesk.ticket" record. @@ -8,22 +10,41 @@ import { getAccessToken } from "src/services/odoo_auth"; export class Ticket { id: number; name: string; + stageName: string; /** * Make a RPC call to the Odoo database to create a ticket * and return the ID of the newly created record. */ - static createTicket(partnerId: number, emailBody: string, emailSubject: string): number { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.CREATE_TICKET; + static createTicket(partner: Partner, email: Email): [Ticket, Partner] | null { + const url = + PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + + URLS.CREATE_TICKET; const odooAccessToken = getAccessToken(); + const [attachments, _] = email.getAttachments(); const response = postJsonRpc( url, - { email_body: emailBody, email_subject: emailSubject, partner_id: partnerId }, + { + email_body: email.body, + email_subject: email.subject, + partner_email: partner.email, + partner_id: partner.id, + partner_name: partner.name, + attachments, + }, { Authorization: "Bearer " + odooAccessToken }, ); - return response ? response.ticket_id || null : null; + if (!response?.id) { + return null; + } + if (!partner.id) { + partner.id = response.partner_id; + partner.image = response.partner_image; + partner.isWritable = true; + } + return [Ticket.fromOdooResponse(response), partner]; } /** @@ -33,6 +54,7 @@ export class Ticket { const ticket = new Ticket(); ticket.id = values.id; ticket.name = values.name; + ticket.stageName = values.stageName; return ticket; } @@ -41,8 +63,9 @@ export class Ticket { */ static fromOdooResponse(values: any): Ticket { const ticket = new Ticket(); - ticket.id = values.ticket_id; + ticket.id = values.id; ticket.name = values.name; + ticket.stageName = values.stage_name; return ticket; } } diff --git a/gmail/src/services/log_email.ts b/gmail/src/services/log_email.ts index 370070843..257f26465 100644 --- a/gmail/src/services/log_email.ts +++ b/gmail/src/services/log_email.ts @@ -12,9 +12,6 @@ import { getAccessToken } from "./odoo_auth"; */ function _formatEmailBody(email: Email, error: ErrorMessage): string { let body = email.body; - - body = `${_t("From:")} ${escapeHtml(email.contactEmail)}

${body}`; - if (error.code === "attachments_size_exceeded") { body += `
${_t( "Attachments could not be logged in Odoo because their total size exceeded the allowed maximum.", @@ -28,8 +25,7 @@ function _formatEmailBody(email: Email, error: ErrorMessage): string { 'class="gmail_chip gmail_drive_chip" style=" min-height: 32px;', ); - body += `

${_t("Logged from")} ${_t("Gmail Inbox")}`; - + body += `

${_t("Logged from")} ${_t("Odoo for Gmail")}`; return body; } @@ -40,11 +36,20 @@ export function logEmail(recordId: number, recordModel: string, email: Email): E const odooAccessToken = getAccessToken(); const [attachments, error] = email.getAttachments(); const body = _formatEmailBody(email, error); - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.LOG_EMAIL; + const url = + PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.LOG_EMAIL; const response = postJsonRpc( url, - { message: body, res_id: recordId, model: recordModel, attachments: attachments }, + { + body, + res_id: recordId, + model: recordModel, + attachments: attachments, + email_from: email.emailFrom, + subject: email.subject, + timestamp: email.timestamp, + }, { Authorization: "Bearer " + odooAccessToken }, ); diff --git a/gmail/src/services/odoo_auth.ts b/gmail/src/services/odoo_auth.ts index fd8bfda47..c4cc613b9 100644 --- a/gmail/src/services/odoo_auth.ts +++ b/gmail/src/services/odoo_auth.ts @@ -1,30 +1,6 @@ import { ODOO_AUTH_URLS } from "../const"; import { postJsonRpc, encodeQueryData } from "../utils/http"; - -const errorPage = ` - - - -
__ERROR_MESSAGE__
-`; +import { RAINBOW, ERROR_PAGE } from "./pages"; /** * Callback function called during the OAuth authentication process. @@ -33,7 +9,7 @@ const errorPage = ` * We generate a state token (for this function) * 2. The user is redirected to Odoo and enter his login / password * 3. Then the user is redirected to the Google App-Script - * 4. Thanks the the state token, the function "odooAuthCallback" is called with the auth code + * 4. Thanks the state token, the function "odooAuthCallback" is called with the auth code * 5. The auth code is exchanged for an access token with a RPC call */ function odooAuthCallback(callbackRequest: any) { @@ -43,7 +19,7 @@ function odooAuthCallback(callbackRequest: any) { if (success !== "1") { return HtmlService.createHtmlOutput( - errorPage.replace("__ERROR_MESSAGE__", "Odoo did not return successfully."), + ERROR_PAGE.replace("__ERROR_MESSAGE__", "Odoo did not return successfully."), ); } @@ -58,7 +34,7 @@ function odooAuthCallback(callbackRequest: any) { if (!response || !response.access_token || !response.access_token.length) { return HtmlService.createHtmlOutput( - errorPage.replace( + ERROR_PAGE.replace( "__ERROR_MESSAGE__", "The token exchange failed. Maybe your token has expired or your database can not be reached by the Google server." + "
Contact your administrator or our support.", @@ -70,14 +46,14 @@ function odooAuthCallback(callbackRequest: any) { userProperties.setProperty("ODOO_ACCESS_TOKEN", accessToken); - return HtmlService.createHtmlOutput("Success !"); + return HtmlService.createHtmlOutput(RAINBOW); } /** * Generate the URL to redirect the user for the authentication to the Odoo database. * * This URL contains a state and the Odoo database should resend it. - * The Google server use the state code to know which function to execute when the user + * The Google server uses the state code to know which function to execute when the user * is redirected on their server. */ export function getOdooAuthUrl() { @@ -93,7 +69,10 @@ export function getOdooAuthUrl() { throw new Error("Can not retrieve the script ID."); } - const stateToken = ScriptApp.newStateToken().withMethod(odooAuthCallback.name).withTimeout(3600).createToken(); + const stateToken = ScriptApp.newStateToken() + .withMethod(odooAuthCallback.name) + .withTimeout(3600) + .createToken(); const redirectToAddon = `https://script.google.com/macros/d/${scriptId}/usercallback`; const scope = ODOO_AUTH_URLS.SCOPE; @@ -132,33 +111,32 @@ export const resetAccessToken = () => { }; /** - * Make an HTTP request to "/mail_plugin/auth/access_token" (cors="*") on the Odoo - * database to verify that the server is reachable and that the mail plugin module is - * installed. + * Make an HTTP request to the Odoo database to verify that the server + * is reachable and that the mail plugin module is installed. * - * Returns True if the Odoo database is reachable and if the "mail_plugin" module - * is installed, false otherwise. + * Returns the version of the addin that is supported if it's reachable, null otherwise. */ -export const isOdooDatabaseReachable = (odooUrl: string): boolean => { +export const getSupportedAddinVersion = (odooUrl: string): number | null => { if (!odooUrl || !odooUrl.length) { - return false; + return null; } const response = postJsonRpc( - odooUrl + ODOO_AUTH_URLS.CODE_VALIDATION, - { auth_code: null }, + odooUrl + ODOO_AUTH_URLS.CHECK_VERSION, + {}, {}, { returnRawResponse: true }, ); if (!response) { - return false; + return null; } const responseCode = response.getResponseCode(); if (responseCode > 299 || responseCode < 200) { - return false; + return null; } - return true; + const textResponse = response.getContentText("UTF-8"); + return parseInt(JSON.parse(textResponse).result); }; diff --git a/gmail/src/services/odoo_redirection.ts b/gmail/src/services/odoo_redirection.ts new file mode 100644 index 000000000..4588e4ab7 --- /dev/null +++ b/gmail/src/services/odoo_redirection.ts @@ -0,0 +1,5 @@ +import { getOdooServerUrl } from "./app_properties"; + +export function getOdooRecordURL(model, record_id) { + return getOdooServerUrl() + `/mail_plugin/redirect_to_record/${model}/?record_id=${record_id}`; +} diff --git a/gmail/src/services/pages.ts b/gmail/src/services/pages.ts new file mode 100644 index 000000000..4b5bdc3ba --- /dev/null +++ b/gmail/src/services/pages.ts @@ -0,0 +1,271 @@ +export const RAINBOW = ` + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + + + + +
+
+
You're all set
You can now close this window and connect with Odoo!
+
+
+
+
+
+ +`; + +export const ERROR_PAGE = ` + + + +
__ERROR_MESSAGE__
+`; diff --git a/gmail/src/services/search_records.ts b/gmail/src/services/search_records.ts new file mode 100644 index 000000000..4ddf6746d --- /dev/null +++ b/gmail/src/services/search_records.ts @@ -0,0 +1,25 @@ +import { postJsonRpc } from "../utils/http"; +import { URLS } from "../const"; +import { ErrorMessage } from "../models/error_message"; +import { _t } from "../services/translation"; +import { getAccessToken } from "./odoo_auth"; + +/** + * Search records of the given model. + */ +export function searchRecords(recordModel: string, query: string): [any[], number, ErrorMessage] { + const odooAccessToken = getAccessToken(); + const url = + PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + + URLS.SEARCH_RECORDS + + "/" + + recordModel; + + const response = postJsonRpc(url, { query }, { Authorization: "Bearer " + odooAccessToken }); + + if (!response?.length) { + return [[], 0, new ErrorMessage("unknown", response.error)]; + } + + return [response[0], response[1], new ErrorMessage(null)]; +} diff --git a/gmail/src/services/translation.ts b/gmail/src/services/translation.ts index 663096eba..ef6d07844 100644 --- a/gmail/src/services/translation.ts +++ b/gmail/src/services/translation.ts @@ -4,7 +4,7 @@ import { getAccessToken } from "./odoo_auth"; import { getOdooServerUrl } from "./app_properties"; /** - * Object which fetchs the translations on the Odoo database, puts them in cache. + * Object which fetch the translations on the Odoo database, puts them in cache. * * Done in a class and not in a simple function so we read only once the cache for all * translations. @@ -32,7 +32,7 @@ export class Translate { ); if (this.translations) { - // Put in the cacher for 6 hours (maximum cache life time) + // Put in the cache for 6 hours (maximum cache lifetime) cache.put(cacheKey, JSON.stringify(this.translations), 21600); } } @@ -69,7 +69,10 @@ export class Translate { .join("|"), "gi", ); - return translated.replace(re, (key) => parameters[key.substring(2, key.length - 2)] || ""); + return translated.replace( + re, + (key) => parameters[key.substring(2, key.length - 2)] || "", + ); } } } diff --git a/gmail/src/utils/format.ts b/gmail/src/utils/format.ts index 23f20089b..5ffc28289 100644 --- a/gmail/src/utils/format.ts +++ b/gmail/src/utils/format.ts @@ -13,36 +13,6 @@ export function formatUrl(url: string): string { return url.replace(/\/+$/, ""); } -/** - * Return the given string only if it's not null and not empty. - */ -export function isTrue(value: any): string { - if (value && value.length) { - return value; - } -} - -/** - * Return the first element of an array if the array is not null and not empty. - */ -export function first(value: any[]): any { - if (value && value.length) { - return value[0]; - } -} - -/** - * Repeat the given string "n" times. - */ -export function repeat(str: string, n: number) { - let result = ""; - while (n > 0) { - result += str; - n--; - } - return result; -} - /** * Truncate the given text to not exceed the given length. */ diff --git a/gmail/src/utils/http.ts b/gmail/src/utils/http.ts index 79a2a34da..b2640d806 100644 --- a/gmail/src/utils/http.ts +++ b/gmail/src/utils/http.ts @@ -4,6 +4,13 @@ import { State } from "../models/state"; * Make a JSON RPC call with the following parameters. */ export function postJsonRpc(url: string, data = {}, headers = {}, options: any = {}) { + for (const key in data) { + // don't send null values + if (data[key] === undefined || data[key] === null) { + data[key] = false; + } + } + // Make a valid "Odoo RPC" call data = { id: 0, @@ -40,46 +47,12 @@ export function postJsonRpc(url: string, data = {}, headers = {}, options: any = } return dictResponse.result; - } catch { + } catch (e) { + Logger.log(`HTTP Error: ${e}`); return; } } -/** - * Make a JSON RPC call with the following parameters. - * - * Try to first read the response from the cache, if not found, - * make the call and cache the response. - * - * The cache key is based on the URL and the JSON data - * - * Store the result for 6 hours by default (maximum cache duration) - * - * This cache may be needed to make to many HTTP call to an external service (e.g. IAP). - */ -export function postJsonRpcCached(url: string, data = {}, headers = {}, cacheTtl: number = 21600) { - const cache = CacheService.getUserCache(); - - // Max 250 characters, to hash the key to have a fixed length - const cacheKey = - "ODOO_HTTP_CACHE_" + - Utilities.base64Encode(Utilities.computeDigest(Utilities.DigestAlgorithm.SHA_256, JSON.stringify([url, data]))); - - const cachedResponse = cache.get(cacheKey); - - if (cachedResponse) { - return JSON.parse(cachedResponse); - } - - const response = postJsonRpc(url, data, headers); - - if (response) { - cache.put(cacheKey, JSON.stringify(response), cacheTtl); - } - - return response; -} - /** * Take a dictionary and return the URL encoded parameters */ diff --git a/gmail/src/views/card_actions.ts b/gmail/src/views/card_actions.ts index d171a5787..304422304 100644 --- a/gmail/src/views/card_actions.ts +++ b/gmail/src/views/card_actions.ts @@ -1,43 +1,29 @@ import { buildDebugView } from "./debug"; -import { buildView } from "../views/index"; import { State } from "../models/state"; -import { Partner } from "../models/partner"; import { resetAccessToken } from "../services/odoo_auth"; import { _t, clearTranslationCache } from "../services/translation"; import { actionCall } from "./helpers"; import { pushToRoot } from "./helpers"; +import { buildLoginMainView } from "../views/login"; -function onLogout(state: State) { +function onLogout() { resetAccessToken(); clearTranslationCache(); - - const [partner, odooUserCompanies, canCreatePartner, canCreateProject, error] = Partner.enrichPartner( - state.email.contactEmail, - state.email.contactName, - ); - const newState = new State( - partner, - canCreatePartner, - state.email, - odooUserCompanies, - null, - null, - canCreateProject, - error, - ); - return pushToRoot(buildView(newState)); + return pushToRoot(buildLoginMainView()); } -export function buildCardActionsView(state: State, card: Card) { - const canContactOdooDatabase = state.error.canContactOdooDatabase && State.isLogged; - +export function buildCardActionsView(card: Card) { if (State.isLogged) { card.addCardAction( - CardService.newCardAction().setText(_t("Logout")).setOnClickAction(actionCall(state, onLogout.name)), + CardService.newCardAction() + .setText(_t("Log out")) + .setOnClickAction(actionCall(undefined, onLogout.name)), ); } card.addCardAction( - CardService.newCardAction().setText(_t("Debug")).setOnClickAction(actionCall(state, buildDebugView.name)), + CardService.newCardAction() + .setText(_t("Debug")) + .setOnClickAction(actionCall(undefined, buildDebugView.name)), ); } diff --git a/gmail/src/views/company.ts b/gmail/src/views/company.ts deleted file mode 100644 index 9a13ef072..000000000 --- a/gmail/src/views/company.ts +++ /dev/null @@ -1,231 +0,0 @@ -import { buildView } from "./index"; -import { actionCall, createKeyValueWidget, notify, updateCard } from "./helpers"; -import { SOCIAL_MEDIA_ICONS, UI_ICONS } from "./icons"; -import { URLS } from "../const"; -import { getOdooServerUrl } from "src/services/app_properties"; -import { ErrorMessage } from "../models/error_message"; -import { State } from "../models/state"; -import { Company } from "../models/company"; -import { Partner } from "../models/partner"; -import { _t } from "../services/translation"; - -/** - * Update the application state with the new company created / enriched. - * IT could be necessary to also update the contact if the contact is the company itself. - */ -function _setContactCompany(state: State, company: Company, error: ErrorMessage) { - if (company) { - state.partner.company = company; - if (state.partner.id === company.id) { - // The contact is the same partner as the company - // update his information - state.partner.isCompany = true; - state.partner.image = company.image; - state.partner.phone = company.phone; - state.partner.mobile = company.mobile; - } - } - state.error = error; - return updateCard(buildView(state)); -} - -function onCreateCompany(state: State) { - const [company, error] = Partner.createCompany(state.partner.id); - return _setContactCompany(state, company, error); -} - -function onEnrichCompany(state: State) { - const [company, error] = Partner.enrichCompany(state.partner.company.id); - return _setContactCompany(state, company, error); -} - -function onUnfoldCompanyDescription(state: State) { - state.isCompanyDescriptionUnfolded = true; - return updateCard(buildView(state)); -} - -export function buildCompanyView(state: State, card: Card) { - if (state.partner.company) { - const odooServerUrl = getOdooServerUrl(); - const cids = state.odooCompaniesParameter; - const company = state.partner.company; - - const companySection = CardService.newCardSection().setHeader("" + _t("Company Insights") + ""); - - if (!state.partner.id || state.partner.id !== company.id) { - const companyContent = [company.email, company.phone] - .filter((x) => x) - .map((x) => `${x}`); - - companySection.addWidget( - createKeyValueWidget( - null, - company.name + "
" + companyContent.join("
"), - company.image || UI_ICONS.no_company, - null, - null, - company.id ? odooServerUrl + `/web#id=${company.id}&model=res.partner&view_type=form${cids}` : null, - false, - company.email, - CardService.ImageCropType.CIRCLE, - ), - ); - } - - _addSocialButtons(companySection, company); - - if (company.description) { - const MAX_DESCRIPTION_LENGTH = 70; - if (company.description.length < MAX_DESCRIPTION_LENGTH || state.isCompanyDescriptionUnfolded) { - companySection.addWidget(createKeyValueWidget(_t("Description"), company.description)); - } else { - companySection.addWidget( - createKeyValueWidget( - _t("Description"), - company.description.substring(0, MAX_DESCRIPTION_LENGTH) + - "..." + - "
" + - "" + - _t("Read more") + - "", - null, - null, - null, - actionCall(state, onUnfoldCompanyDescription.name), - ), - ); - } - } - - if (company.address) { - companySection.addWidget( - createKeyValueWidget( - _t("Address"), - company.address, - UI_ICONS.location, - null, - null, - "https://www.google.com/maps/search/" + encodeURIComponent(company.address).replace("/", " "), - ), - ); - } - - if (company.phones) { - companySection.addWidget(createKeyValueWidget(_t("Phones"), company.phones, UI_ICONS.phone)); - } - - if (company.website) { - companySection.addWidget( - createKeyValueWidget(_t("Website"), company.website, UI_ICONS.website, null, null, company.website), - ); - } - - if (company.industry) { - companySection.addWidget(createKeyValueWidget(_t("Industry"), company.industry, UI_ICONS.industry)); - } - - if (company.employees) { - companySection.addWidget( - createKeyValueWidget(_t("Employees"), _t("%s employees", company.employees), UI_ICONS.people), - ); - } - - if (company.foundedYear) { - companySection.addWidget( - createKeyValueWidget(_t("Founded Year"), "" + company.foundedYear, UI_ICONS.foundation), - ); - } - - if (company.tags) { - companySection.addWidget(createKeyValueWidget(_t("Keywords"), company.tags, UI_ICONS.keywords)); - } - - if (company.companyType) { - companySection.addWidget( - createKeyValueWidget(_t("Company Type"), company.companyType, UI_ICONS.company_type), - ); - } - - if (company.annualRevenue) { - companySection.addWidget(createKeyValueWidget(_t("Annual Revenue"), company.annualRevenue, UI_ICONS.money)); - } - - card.addSection(companySection); - - if (!company.isEnriched) { - const enrichSection = CardService.newCardSection(); - enrichSection.addWidget(CardService.newTextParagraph().setText(_t("No insights for this company."))); - if (state.error.canCreateCompany && state.canCreatePartner) { - enrichSection.addWidget( - CardService.newTextButton() - .setText(_t("Enrich Company")) - .setOnClickAction(actionCall(state, onEnrichCompany.name)), - ); - } - card.addSection(enrichSection); - } - } else if (state.partner.id) { - const companySection = CardService.newCardSection().setHeader("" + _t("Company Insights") + ""); - companySection.addWidget(CardService.newTextParagraph().setText(_t("No company attached to this contact."))); - - if (state.error.canCreateCompany && state.canCreatePartner) { - companySection.addWidget( - CardService.newTextButton() - .setText(_t("Create a company")) - .setOnClickAction(actionCall(state, onCreateCompany.name)), - ); - } - card.addSection(companySection); - } -} - -function _addSocialButtons(section: CardSection, company: Company) { - const socialMediaButtons = CardService.newButtonSet(); - - const socialMedias = [ - { - name: "Facebook", - url: "https://facebook.com/", - icon: SOCIAL_MEDIA_ICONS.facebook, - key: "facebook", - }, - { - name: "Twitter", - url: "https://twitter.com/", - icon: SOCIAL_MEDIA_ICONS.twitter, - key: "twitter", - }, - { - name: "LinkedIn", - url: "https://linkedin.com/", - icon: SOCIAL_MEDIA_ICONS.linkedin, - key: "linkedin", - }, - { - name: "Github", - url: "https://github.com/", - icon: SOCIAL_MEDIA_ICONS.github, - key: "github", - }, - { - name: "Crunchbase", - url: "https://crunchbase.com/", - icon: SOCIAL_MEDIA_ICONS.crunchbase, - key: "crunchbase", - }, - ]; - - for (let media of socialMedias) { - const url = company[media.key]; - if (url && url.length) { - socialMediaButtons.addButton( - CardService.newImageButton() - .setAltText(media.name) - .setIconUrl(media.icon) - .setOpenLink(CardService.newOpenLink().setUrl(media.url + url)), - ); - } - } - - section.addWidget(socialMediaButtons); -} diff --git a/gmail/src/views/create_task.ts b/gmail/src/views/create_task.ts index af301468c..5c4cf04e5 100644 --- a/gmail/src/views/create_task.ts +++ b/gmail/src/views/create_task.ts @@ -2,34 +2,31 @@ import { buildView } from "../views/index"; import { updateCard, pushCard, pushToRoot } from "./helpers"; import { UI_ICONS } from "./icons"; import { createKeyValueWidget, actionCall, notify } from "./helpers"; -import { URLS } from "../const"; import { getOdooServerUrl } from "src/services/app_properties"; -import { ErrorMessage } from "../models/error_message"; import { Project } from "../models/project"; import { State } from "../models/state"; import { Task } from "../models/task"; -import { logEmail } from "../services/log_email"; import { _t } from "../services/translation"; +import { getOdooRecordURL } from "src/services/odoo_redirection"; function onSearchProjectClick(state: State, parameters: any, inputs: any) { - const inputQuery = inputs.search_project_query; - const query = (inputQuery && inputQuery.length && inputQuery[0]) || ""; + const query = inputs.search_project_query || ""; const [projects, error] = Project.searchProject(query); + if (error.code) { + return notify(error.message); + } - state.error = error; state.searchedProjects = projects; + return updateCard(buildCreateTaskView(state, query)); +} - const createTaskView = buildCreateTaskView(state, query, true); - - // If go back, show again the "Create Project" section, but do not show all old searches - return parameters.hideCreateProjectSection ? updateCard(createTaskView) : pushCard(createTaskView); +function onCreateProjectViewClick(state: State, parameters: any, inputs: any) { + return updateCard(buildCreateProjectView(state)); } function onCreateProjectClick(state: State, parameters: any, inputs: any) { - const inputQuery = inputs.new_project_name; - const projectName = (inputQuery && inputQuery.length && inputQuery[0]) || ""; - - if (!projectName || !projectName.length) { + const projectName = inputs.new_project_name || ""; + if (!projectName.length) { return notify(_t("The project name is required")); } @@ -43,37 +40,34 @@ function onCreateProjectClick(state: State, parameters: any, inputs: any) { function onSelectProject(state: State, parameters: any) { const project = Project.fromJson(parameters.project); - const task = Task.createTask(state.partner.id, project.id, state.email.body, state.email.subject); + const result = Task.createTask(state.partner, project.id, state.email); - if (!task) { + if (!result) { return notify(_t("Could not create the task")); } - task.projectName = project.name; + const [task, partner] = result; + state.partner = partner; state.partner.tasks.push(task); + state.partner.taskCount += 1; - const taskUrl = - PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + - `/web#id=${task.id}&action=project_mail_plugin.project_task_action_form_edit&model=project.task&view_type=form`; - - // Open the URL to the Odoo task and update the card - return CardService.newActionResponseBuilder() - .setOpenLink(CardService.newOpenLink().setUrl(taskUrl)) - .setNavigation(pushToRoot(buildView(state))) - .build(); + const taskUrl = getOdooRecordURL("project.task", task.id); + return pushToRoot(buildView(state)); } -export function buildCreateTaskView(state: State, query: string = "", hideCreateProjectSection: boolean = false) { +export function buildCreateTaskView(state: State, query: string = "") { let noProject = false; if (!state.searchedProjects) { // Initiate the search - [state.searchedProjects, state.error] = Project.searchProject(""); + const [searchedProjects, error] = Project.searchProject(""); + if (error.code) { + return notify(error.message); + } + + state.searchedProjects = searchedProjects; noProject = !state.searchedProjects.length; } - const odooServerUrl = getOdooServerUrl(); - const partner = state.partner; - const tasks = partner.tasks; const projects = state.searchedProjects; const card = CardService.newCardBuilder(); @@ -88,32 +82,37 @@ export function buildCreateTaskView(state: State, query: string = "", hideCreate .setFieldName("search_project_query") .setTitle(_t("Search a Project")) .setValue(query || "") - .setOnChangeAction( - actionCall(state, onSearchProjectClick.name, { - hideCreateProjectSection: hideCreateProjectSection, - }), - ), + .setOnChangeAction(actionCall(state, onSearchProjectClick.name, {})), ); - projectSection.addWidget( + const actionButtonSet = CardService.newButtonSet(); + actionButtonSet.addButton( CardService.newTextButton() .setText(_t("Search")) - .setOnClickAction( - actionCall(state, onSearchProjectClick.name, { - hideCreateProjectSection: hideCreateProjectSection, - }), - ), + .setOnClickAction(actionCall(state, onSearchProjectClick.name, {})), ); + if (state.canCreateProject) { + actionButtonSet.addButton( + CardService.newTextButton() + .setText(_t("Create Project")) + .setBackgroundColor("#875a7b") + .setOnClickAction(actionCall(state, onCreateProjectViewClick.name, {})), + ); + } + projectSection.addWidget(actionButtonSet); if (!projects.length) { - projectSection.addWidget(CardService.newTextParagraph().setText(_t("No project found."))); + projectSection.addWidget( + CardService.newTextParagraph().setText(_t("No project found.")), + ); } for (let project of projects) { + const bottomLabel = [project.companyName, project.partnerName, project.stageName]; const projectCard = createKeyValueWidget( null, project.name, null, - project.partnerName, + bottomLabel.filter((l) => l).join(" - "), null, actionCall(state, onSelectProject.name, { project: project }), ); @@ -121,33 +120,22 @@ export function buildCreateTaskView(state: State, query: string = "", hideCreate projectSection.addWidget(projectCard); } card.addSection(projectSection); - } - - if (!hideCreateProjectSection && state.canCreateProject) { - const createProjectSection = CardService.newCardSection().setHeader( - "" + _t("Create a Task in a new Project") + "", - ); - - createProjectSection.addWidget( - CardService.newTextInput().setFieldName("new_project_name").setTitle(_t("Project Name")).setValue(""), - ); - - createProjectSection.addWidget( - CardService.newTextButton() - .setText(_t("Create Project & Task")) - .setOnClickAction(actionCall(state, onCreateProjectClick.name)), - ); - card.addSection(createProjectSection); - } else if (noProject) { + } else if (state.canCreateProject) { + return buildCreateProjectView(state); + } else { const noProjectSection = CardService.newCardSection(); noProjectSection.addWidget(CardService.newImage().setImageUrl(UI_ICONS.empty_folder)); - noProjectSection.addWidget(CardService.newTextParagraph().setText("" + _t("No project") + "")); + noProjectSection.addWidget( + CardService.newTextParagraph().setText("" + _t("No project") + ""), + ); noProjectSection.addWidget( CardService.newTextParagraph().setText( - _t("There are no project in your database. Please ask your project manager to create one."), + _t( + "There are no project in your database. Please ask your project manager to create one.", + ), ), ); @@ -156,3 +144,27 @@ export function buildCreateTaskView(state: State, query: string = "", hideCreate return card.build(); } + +export function buildCreateProjectView(state: State) { + const card = CardService.newCardBuilder(); + + const createProjectSection = CardService.newCardSection().setHeader( + "" + _t("Create a Task in a new Project") + "", + ); + + createProjectSection.addWidget( + CardService.newTextInput() + .setFieldName("new_project_name") + .setTitle(_t("Project Name")) + .setValue(""), + ); + + createProjectSection.addWidget( + CardService.newTextButton() + .setText(_t("Create Project & Task")) + .setOnClickAction(actionCall(state, onCreateProjectClick.name)), + ); + card.addSection(createProjectSection); + + return card.build(); +} diff --git a/gmail/src/views/debug.ts b/gmail/src/views/debug.ts index 4576952e8..9fd9c2b8c 100644 --- a/gmail/src/views/debug.ts +++ b/gmail/src/views/debug.ts @@ -9,20 +9,30 @@ export function buildDebugView() { const odooAccessToken = getAccessToken(); card.setHeader( - CardService.newCardHeader().setTitle(_t("Debug Zone")).setSubtitle(_t("Debug zone for development purpose.")), + CardService.newCardHeader() + .setTitle(_t("Debug Zone")) + .setSubtitle(_t("Debug zone for development purpose.")), ); - card.addSection(CardService.newCardSection().addWidget(createKeyValueWidget(_t("Odoo Server URL"), odooServerUrl))); + card.addSection( + CardService.newCardSection().addWidget( + createKeyValueWidget(_t("Odoo Server URL"), odooServerUrl), + ), + ); card.addSection( - CardService.newCardSection().addWidget(createKeyValueWidget(_t("Odoo Access Token"), odooAccessToken)), + CardService.newCardSection().addWidget( + createKeyValueWidget(_t("Odoo Access Token"), odooAccessToken), + ), ); card.addSection( CardService.newCardSection().addWidget( CardService.newTextButton() .setText(_t("Clear Translations Cache")) - .setOnClickAction(CardService.newAction().setFunctionName(clearTranslationCache.name)), + .setOnClickAction( + CardService.newAction().setFunctionName(clearTranslationCache.name), + ), ), ); diff --git a/gmail/src/views/error.ts b/gmail/src/views/error.ts deleted file mode 100644 index cc4af4baa..000000000 --- a/gmail/src/views/error.ts +++ /dev/null @@ -1,63 +0,0 @@ -import { State } from "../models/state"; -import { createKeyValueWidget, actionCall } from "./helpers"; -import { buildView } from "./index"; -import { updateCard } from "./helpers"; -import { UI_ICONS } from "./icons"; -import { _t } from "../services/translation"; -import { buildLoginMainView } from "./login"; - -function onCloseError(state: State) { - state.error.code = null; - return updateCard(buildView(state)); -} - -function _addError(message: string, state: State, icon: string = null): CardSection { - const errorSection = CardService.newCardSection(); - - errorSection.addWidget( - createKeyValueWidget( - null, - message, - icon, - null, - CardService.newImageButton() - .setAltText(_t("Close")) - .setIconUrl(UI_ICONS.close) - .setOnClickAction(actionCall(state, onCloseError.name)), - ), - ); - return errorSection; -} - -export function buildErrorView(state: State, card: Card) { - const error = state.error; - - const ignoredErrors = ["company_created", "company_updated"]; - if (ignoredErrors.indexOf(error.code) >= 0) { - return; - } - - if (error.code === "http_error_odoo") { - const errorSection = _addError(error.message, state); - errorSection.addWidget( - CardService.newTextButton() - .setText(_t("Login")) - .setOnClickAction(CardService.newAction().setFunctionName(buildLoginMainView.name)), - ); - card.addSection(errorSection); - } else if (error.code === "insufficient_credit") { - const errorSection = _addError(error.message, state); - errorSection.addWidget( - CardService.newTextButton() - .setText(_t("Buy new credits")) - .setOpenLink(CardService.newOpenLink().setUrl(error.information)), - ); - card.addSection(errorSection); - } else if (error.code === "missing_data") { - card.addSection(_addError(error.message, state)); - } else { - let errors = [error.message, error.information].filter((x) => x); - const errorMessage = errors.join("\n"); - card.addSection(_addError(errorMessage, state)); - } -} diff --git a/gmail/src/views/helpers.ts b/gmail/src/views/helpers.ts index db715e840..3cb705c11 100644 --- a/gmail/src/views/helpers.ts +++ b/gmail/src/views/helpers.ts @@ -1,6 +1,7 @@ import { UI_ICONS } from "./icons"; import { State } from "../models/state"; import { escapeHtml } from "../utils/html"; +import { truncate } from "../utils/format"; /** * Remove all cards and push the new one @@ -26,7 +27,7 @@ export function pushCard(card: Card) { /** * Build a widget "Key / Value / Icon" * - * If the icon if not a valid URL, take the icon from: + * If the icon is not a valid URL, take the icon from: * https://github.com/webdog/octicons-png */ export function createKeyValueWidget( @@ -40,7 +41,7 @@ export function createKeyValueWidget( iconLabel: string = null, iconCropStyle: GoogleAppsScript.Card_Service.ImageCropType = CardService.ImageCropType.SQUARE, ) { - const widget = CardService.newDecoratedText().setText(content).setWrapText(true); + const widget = CardService.newDecoratedText().setText(content); if (label && label.length) { widget.setTopLabel(escapeHtml(label)); } @@ -62,7 +63,9 @@ export function createKeyValueWidget( if (icon && icon.length) { const isIconUrl = - icon.indexOf("http://") === 0 || icon.indexOf("https://") === 0 || icon.indexOf("data:image/") === 0; + icon.indexOf("http://") === 0 || + icon.indexOf("https://") === 0 || + icon.indexOf("data:image/") === 0; if (!isIconUrl) { throw new Error("Invalid icon URL"); } @@ -82,10 +85,12 @@ export function createKeyValueWidget( function _handleActionCall(event) { const functionName = event.parameters.functionName; - const state = State.fromJson(event.parameters.state); const parameters = JSON.parse(event.parameters.parameters); - const inputs = event.formInputs; - return eval(functionName)(state, parameters, inputs); + if (event.parameters.state?.length) { + const state = State.fromJson(event.parameters.state); + return eval(functionName)(state, parameters, event.formInput); + } + return eval(functionName)(parameters, event.formInput); } /** @@ -95,12 +100,12 @@ function _handleActionCall(event) { * must be strings. Therefor we serialized the state and other arguments to clean the code * and to be able to access to it in the event handlers. */ -export function actionCall(state: State, functionName: string, parameters: any = {}) { +export function actionCall(state: State | null, functionName: string, parameters: any = {}) { return CardService.newAction() .setFunctionName(_handleActionCall.name) .setParameters({ functionName: functionName, - state: state.toJson(), + state: state ? state.toJson() : "", parameters: JSON.stringify(parameters), }); } @@ -112,5 +117,7 @@ export function notify(message: string) { } export function openUrl(url: string) { - return CardService.newActionResponseBuilder().setOpenLink(CardService.newOpenLink().setUrl(url)).build(); + return CardService.newActionResponseBuilder() + .setOpenLink(CardService.newOpenLink().setUrl(url)) + .build(); } diff --git a/gmail/src/views/icons.ts b/gmail/src/views/icons.ts index ef95aecf1..d45e6aab7 100644 --- a/gmail/src/views/icons.ts +++ b/gmail/src/views/icons.ts @@ -1,91 +1,23 @@ -// Icon come from https://www.iconfinder.com/ -// Store as PNG 64x64 - -export const SOCIAL_MEDIA_ICONS = { - facebook: - "", - twitter: - "", - github: - "", - linkedin: - "", - crunchbase: - "", -}; - export const UI_ICONS = { - person: - "", - phone: - "", - home: - "", - people: - "", - project: - "", - work: - "", - money: - "", - interrogation: - "", - industry: - "", - twitter: - "", - timezone: - "", - keywords: - "", - company_type: - "", - email: - "", - odoo: - "", - foundation: - "", - location: - "", - search: - "", - website: - "", - no_result: - "", - save_in_odoo: - "", - open_in_odoo: - "", + person: "", + odoo: "", email_in_odoo: "", email_logged: "", - reload: - "", - close: - "", - check: - "", - no_company: - "", + reload: "", + close: "", empty_folder: "", + search: "", + no_record: + "PHN2ZyB3aWR0aD0iNjQiIGhlaWdodD0iNjQiIHZpZXdCb3g9Ii0yMCAwIDEwNCA5MCIgZmlsbD0ibm9uZSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KPHBhdGggZD0iTTI4LjAzNjYgNDIuMjMyMkMyMC42MTUxIDM4LjY5NTkgMTguODY1IDI2Ljg3NTkgMjQuMTI3NiAxNS44MzE0QzI5LjM5MDIgNC43ODY4NyAzOS42NzI3IC0xLjI5OTc4IDQ3LjA5NDEgMi4yMzY0N0w1MS41ODMyIDMuOTU1MzVMMzMuMjQ4NCA0NC45NjE1TDI4LjAzNjYgNDIuMjMyMloiIGZpbGw9IiNGQkRCRDAiLz4KPHBhdGggZD0iTTkuNjQ5MDEgNjIuNDI3OEM5LjYwODcyIDYyLjQyNzggOS41NjgyMSA2Mi40MTQ0IDkuNTM0NSA2Mi4zODcyTDUuMjU1NiA1OC45MjM2QzUuMjE3MzEgNTguODkyNiA1LjE5MzIyIDU4Ljg0NzUgNS4xODg4NSA1OC43OTgzQzUuMTg0NCA1OC43NDkyIDUuMjAwMTIgNTguNzAwNiA1LjIzMjI2IDU4LjY2MzNDNS44MzAwMiA1Ny45NjkxIDE5Ljg3ODQgNDEuNjU5NyAyMC4yNjYgNDEuMzI4M0MyMC41Mjc4IDQxLjEwNDEgMjAuODIyNiA0MC45NzM4IDIxLjE0MjkgNDAuOTQwNUMyMS4xNTUgNDAuOTI2NyAyMS4xNjk0IDQwLjkxNDYgMjEuMTg1OCA0MC45MDVDMjEuMjE1MyA0MC44ODc0IDIxLjI0ODEgNDAuODc4OSAyMS4yODA3IDQwLjg3ODlDMjEuMjkzNSA0MC44Nzg5IDIxLjMwNjQgNDAuODgwMyAyMS4zMTkgNDAuODgyOUMyMS4zNzY2IDQwLjg5MDEgMjEuNDM0MyA0MC44OTYxIDIxLjQ5MjEgNDAuOTAyMUMyMS42MzQ3IDQwLjkxNzMgMjEuNzgyNCA0MC45MzI5IDIxLjkyOSA0MC45NjIyQzIyLjc3MzcgNDEuMTMxNCAyMy41MjY4IDQxLjU2MTkgMjQuMjMxNCA0Mi4yNzgyTDI0LjI3MDUgNDIuMzE4QzI0Ljc0OTggNDIuODA0MyAyNS4zNDYzIDQzLjQwOTcgMjUuMzQ2MyA0NC42MjAyQzI1LjM0NjMgNDQuNjM2NyAyNS4zNDQgNDQuNjUzMiAyNS4zMzk1IDQ0LjY2OTNDMjUuMjY3NCA0NC45MjczIDE1LjU2MzMgNTUuODY4IDkuNzg1MTUgNjIuMzY2NkM5Ljc0OTMzIDYyLjQwNyA5LjY5OTM1IDYyLjQyNzggOS42NDkwMSA2Mi40Mjc4WiIgZmlsbD0id2hpdGUiLz4KPHBhdGggZD0iTTIxLjI4MjkgNDEuMDU5MUgyMS4yODQxSDIxLjI4MjlaTTIxLjI4MjkgNDEuMDU5MUwyMS4yODA0IDQxLjA2MTZDMjEuMjc5OSA0MS4wNjE1IDIxLjI3OTMgNDEuMDYxNCAyMS4yNzg4IDQxLjA2MTRMMjEuMjgyOSA0MS4wNTkxWk0yMS4yODA0IDQxLjA2MTZDMjEuNDg0OCA0MS4wODcyIDIxLjY5MTcgNDEuMTAwNCAyMS44OTMyIDQxLjE0MDhDMjIuNzY4OSA0MS4zMTYzIDIzLjQ4NjEgNDEuNzgwMyAyNC4xMDE1IDQyLjQwNTlDMjQuNTkwOSA0Mi45MDM0IDI1LjE2NDEgNDMuNDUzMiAyNS4xNjQxIDQ0LjYyMDJDMjUuMDkyNiA0NC44NzYyIDkuNjQ5MDQgNjIuMjQ1NyA5LjY0OTA0IDYyLjI0NTdMNS4zNzAwOSA1OC43ODJDNS4zNzAwOSA1OC43ODIgMjAuMDAyNyA0MS43OTMgMjAuMzg0MyA0MS40NjY1QzIwLjYyNDkgNDEuMjYwNiAyMC45MDQxIDQxLjEzNTEgMjEuMjI1MiA0MS4xMTYzQzIxLjI0MzYgNDEuMDk4IDIxLjI2MiA0MS4wNzk4IDIxLjI4MDQgNDEuMDYxNlpNMjEuMjg0IDQwLjY5NDhIMjEuMjgyOUMyMS4yMDY1IDQwLjY5NDggMjEuMTM1NSA0MC43MTg0IDIxLjA3NjkgNDAuNzU4NkMyMS4wNzM0IDQwLjc2MTEgMjEuMDY5OSA0MC43NjM1IDIxLjA2NjQgNDAuNzY2MUMyMC43MzEyIDQwLjgxMjIgMjAuNDIyNiA0MC45NTQ0IDIwLjE0NzQgNDEuMTg5OEMxOS43NjQzIDQxLjUxNzcgNy41NDA2MyA1NS43MDM4IDUuMDk0MTEgNTguNTQ0M0M1LjAyOTgyIDU4LjYxODkgNC45OTg1IDU4LjcxNjUgNS4wMDczMiA1OC44MTQ2QzUuMDE2MTUgNTguOTEyNyA1LjA2NDMzIDU5LjAwMzEgNS4xNDA5MiA1OS4wNjUxTDkuNDE5ODcgNjIuNTI4OEM5LjQ4NzE2IDYyLjU4MzMgOS41NjgyNCA2Mi42MSA5LjY0ODg4IDYyLjYxQzkuNzQ5NSA2Mi42MSA5Ljg0OTQ1IDYyLjU2ODQgOS45MjEyMiA2Mi40ODc3QzkuOTU5ODMgNjIuNDQ0MyAxMy44MjY3IDU4LjA5NSAxNy42NTI1IDUzLjc3MDNDMjUuNDQ3NiA0NC45NTg5IDI1LjQ3ODggNDQuODQ3MyAyNS41MTQ5IDQ0LjcxODJDMjUuNTIzOCA0NC42ODYzIDI1LjUyODMgNDQuNjUzNCAyNS41MjgzIDQ0LjYyMDJDMjUuNTI4MyA0My4zMzQ4IDI0LjkwMjggNDIuNzAwMSAyNC40MDAxIDQyLjE5TDI0LjM2MTIgNDIuMTUwNUMyMy42MzAyIDQxLjQwNzQgMjIuODQ2MyA0MC45NjAzIDIxLjk2NDggNDAuNzgzN0MyMS44MTAxIDQwLjc1MjcgMjEuNjU4NiA0MC43MzY2IDIxLjUxMiA0MC43MjExQzIxLjQ2NTkgNDAuNzE2MiAyMS40MTk3IDQwLjcxMTMgMjEuMzczNyA0MC43MDU5QzIxLjM0NSA0MC42OTg3IDIxLjMxNSA0MC42OTQ4IDIxLjI4NCA0MC42OTQ4WiIgZmlsbD0iIzM3NDg3NCIvPgo8cGF0aCBkPSJNMjQuOTYzOSA0NC4yMDU1QzI0Ljk1OTUgNDMuNDcwMiAyNC41OTA5IDQyLjkwMzQgMjQuMTAxNSA0Mi40MDU5QzIzLjQ4NjEgNDEuNzgwMyAyMi43NjkgNDEuMzE2MyAyMS44OTMyIDQxLjE0MDhDMjEuNjkxMiA0MS4xMDA0IDIxLjQ4MzggNDEuMDg3MiAyMS4yNzg4IDQxLjA2MTRDMjEuMjc4OCA0MS4wNjE0IDIxLjM1NDQgNDAuOTE1NiAyMS40MDY0IDQwLjg1NTRDMjIuMTU1NCAzOS45ODc2IDIyLjkwNzMgMzkuMTIyMyAyMy42NTg1IDM4LjI1NjRDMjMuOTI5NCAzOC4wNDMxIDI0LjIyMzUgMzcuOTA2MSAyNC41ODA0IDM3Ljg5NDJDMjUuMjM5OCAzNy44NzIyIDI1LjgwMjcgMzguMTA1MiAyNi4zMDIzIDM4LjUwOTRDMjYuNzUzNCAzOC44NzQ0IDI3LjE1MTQgMzkuMjkyOCAyNy40MDI1IDM5LjgyMzNDMjcuNzA0MyA0MC40NjEgMjcuNjg2NiA0MS4wODMxIDI3LjI1MDYgNDEuNjY1MkMyNy4yNTA2IDQxLjY2NTIgMjcuMTE2MSA0MS43NjU5IDI3LjA2MTYgNDEuODI3NEMyNi4zNjEgNDIuNjE4OCAyNS42NjI4IDQzLjQxMjQgMjQuOTYzOSA0NC4yMDU1WiIgZmlsbD0id2hpdGUiLz4KPHBhdGggZD0iTTM3LjQ3OTUgNDUuNjE4MkMzNS44MzA2IDQ1LjYxOCAzNC4yNzIxIDQ1LjI3MzYgMzIuODQ3MyA0NC41OTQ1QzI5LjIwMTUgNDIuODU3NSAyNi43ODE5IDM5LjEyMDQgMjYuMDM0NSAzNC4wNzIyQzI1LjI5MDkgMjkuMDUwMyAyNi4yOTE2IDIzLjMyNTEgMjguODUyMiAxNy45NTEyQzMzLjA0MzEgOS4xNTU5MSA0MC41MDg1IDMuMjQ2NzcgNDcuNDI5MiAzLjI0Njc3QzQ5LjA3NzcgMy4yNDY3NyA1MC42MzYyIDMuNTkwOTcgNTIuMDYxNCA0LjI3MDE5QzU5LjU2MTYgNy44NDM4NiA2MS4zNTM4IDE5Ljc5NjMgNTYuMDU2NSAzMC45MTM3QzUxLjg2NTggMzkuNzA4NiA0NC40MDA0IDQ1LjYxODIgMzcuNDgxIDQ1LjYxODJMMzcuNDc5NSA0NS42MTgyWiIgZmlsbD0id2hpdGUiLz4KPHBhdGggZD0iTTQ3LjQyOTIgMy40Mjg5M0M0OS4wMTUzIDMuNDI4OTMgNTAuNTUzIDMuNzUzMiA1MS45ODMxIDQuNDM0NjJDNTkuNDA0NiA3Ljk3MDgyIDYxLjE1NDcgMTkuNzkwOSA1NS44OTIxIDMwLjgzNTRDNTEuNjQzNSAzOS43NTE4IDQ0LjEyNDcgNDUuNDM2IDM3LjQ3OTQgNDUuNDM2QzM1Ljg5MzMgNDUuNDM2IDM0LjM1NTYgNDUuMTExNyAzMi45MjU1IDQ0LjQzMDNDMjUuNTA0MSA0MC44OTQxIDIzLjc1NCAyOS4wNzQgMjkuMDE2NSAxOC4wMjk1QzMzLjI2NTEgOS4xMTMxMyA0MC43ODM5IDMuNDI4OTMgNDcuNDI5MiAzLjQyODkzWk00Ny40MjkyIDMuMDY0N0M0NC4wNDUzIDMuMDY0NyA0MC40NzM0IDQuNDU5NjIgMzcuMDk5NyA3LjA5ODY5QzMzLjY4MDcgOS43NzMxOCAzMC43NzE5IDEzLjQ5ODggMjguNjg3NyAxNy44NzI5QzI2LjExMTcgMjMuMjc5MSAyNS4xMDU0IDI5LjA0MTYgMjUuODU0MiAzNC4wOTg5QzI2LjYxMDkgMzkuMjA5MSAyOS4wNjY1IDQyLjk5NSAzMi43Njg5IDQ0Ljc1OTFDMzQuMjE4NyA0NS40NSAzNS44MDM2IDQ1LjgwMDMgMzcuNDc5NCA0NS44MDAzQzQwLjg2MzMgNDUuODAwMyA0NC40MzUyIDQ0LjQwNTMgNDcuODA4OSA0MS43NjYyQzUxLjIyNzkgMzkuMDkxOCA1NC4xMzY3IDM1LjM2NjEgNTYuMjIwOSAzMC45OTIxQzU4Ljc5NjkgMjUuNTg1OCA1OS44MDMyIDE5LjgyMzMgNTkuMDU0NCAxNC43NjZDNTguMjk3NyA5LjY1NTgzIDU1Ljg0MjEgNS44Njk5NSA1Mi4xMzk3IDQuMTA1ODRDNTAuNjg5OSAzLjQxNDk4IDQ5LjEwNSAzLjA2NDcgNDcuNDI5MiAzLjA2NDdaIiBmaWxsPSIjMzc0ODc0Ii8+CjxwYXRoIGQ9Ik01Mi43Mzk1IDI5LjMzMzNDNTYuNzY3NSAyMC44Nzk4IDU1LjQyOCAxMS44MzI4IDQ5Ljc0NzYgOS4xMjYxOEM0NC4wNjczIDYuNDE5NTUgMzYuMTk3MSAxMS4wNzgyIDMyLjE2OTEgMTkuNTMxN0MyOC4xNDEyIDI3Ljk4NTEgMjkuNDgwNyAzNy4wMzIxIDM1LjE2MTEgMzkuNzM4N0M0MC44NDE0IDQyLjQ0NTQgNDguNzExNiAzNy43ODY3IDUyLjczOTUgMjkuMzMzM1oiIGZpbGw9IiNGQkRCRDAiLz4KPHBhdGggZD0iTTQ5LjY4NDYgMjcuNDIwM0M1My4xMjQzIDIwLjIwMTQgNTIuNjQ2OSAxMi41NTMxIDQ4Ljg5NzUgOC43ODY1MkM0My4zMDUgNi44ODk3IDM2IDExLjQ5MTggMzIuMTY5MSAxOS41MzE2QzI4LjcyOTQgMjYuNzUwNSAyOS4yMDY4IDM0LjM5ODggMzIuOTU2MiAzOC4xNjU0QzM4LjU0ODcgNDAuMDYyMyA0NS44NTM3IDM1LjQ2MDEgNDkuNjg0NiAyNy40MjAzWiIgZmlsbD0iI0MxREJGNiIvPgo8cGF0aCBkPSJNMzUuODA1OCAyMC4xOTA5QzM1Ljc5OTMgMTkuNzM5OCAzNS45ODQyIDE5LjQ2MzggMzYuMTI0MSAxOS4xODQ0QzM2LjcxMTQgMTguMDExNiAzNy4zODc4IDE2Ljg5MjUgMzguMTkwOCAxNS44NTA5QzM5LjI5NzkgMTQuNDE0NiA0MC41NDM1IDEzLjEyNTEgNDIuMDU4NyAxMi4xMTI0QzQyLjM1OTIgMTEuOTExNiA0Mi42NDU2IDExLjY4NzggNDIuOTc3NyAxMS41Mzg1QzQzLjI4MzQgMTEuNDAxIDQzLjY0ODYgMTEuNTEzNSA0My43ODkgMTEuNzgyM0M0My45Mzk4IDEyLjA3MTQgNDMuODQ1OCAxMi40MDE5IDQzLjU1NTUgMTIuNTkxM0M0My40NDcyIDEyLjY2MTkgNDMuMzM0MyAxMi43MjUyIDQzLjIyMzcgMTIuNzkyMkM0MS45ODg0IDEzLjU0MiA0MC45NDI2IDE0LjUwODIgMzkuOTgzMiAxNS41ODIxQzM4LjkwMTIgMTYuNzkzMiAzOC4wMjgzIDE4LjE0MDEgMzcuMjkxOSAxOS41ODA4QzM3LjE3MDYgMTkuODE4MiAzNy4wNjY1IDIwLjA2NDUgMzYuOTYwMyAyMC4zMDkyQzM2Ljg0NDUgMjAuNTc2IDM2LjU1MjcgMjAuNzU3OSAzNi4zMDQ4IDIwLjcwOTlDMzYuMDA0NiAyMC42NTE4IDM1LjgwNTEgMjAuMzk4NCAzNS44MDU4IDIwLjE5MDlaIiBmaWxsPSJ3aGl0ZSIvPgo8cGF0aCBkPSJNOS42NDkxMSA2Mi4yNDU4QzEwLjI5NSA2MS40NDc5IDkuODYwNyA2MC4wMjU3IDguNjc5MTEgNTkuMDY5MkM3LjQ5NzUxIDU4LjExMjcgNi4wMTYwNCA1Ny45ODQyIDUuMzcwMTYgNTguNzgyMUM0LjcyNDI4IDU5LjU4IDUuMTU4NTYgNjEuMDAyMiA2LjM0MDE2IDYxLjk1ODdDNy41MjE3NSA2Mi45MTUxIDkuMDAzMjIgNjMuMDQzNyA5LjY0OTExIDYyLjI0NThaIiBmaWxsPSIjRkJEQkQwIi8+CjxwYXRoIGQ9Ik00Mi41NDA0IDEuMjMwNzVDNDQuMTI2NSAxLjIzMDc1IDQ1LjY2NDEgMS41NTUwNyA0Ny4wOTQyIDIuMjM2NDRMNTEuNDY5OCA0LjIwOTExQzUxLjY0MjEgNC4yNzk4IDUxLjgxMzYgNC4zNTM4MiA1MS45ODMxIDQuNDM0NTVDNTkuNDA0NSA3Ljk3MDgyIDYxLjE1NDYgMTkuNzkwOCA1NS44OTIxIDMwLjgzNTRDNTEuNjQzOSAzOS43NTA4IDQ0LjEyNTEgNDUuNDM0NiAzNy40ODA0IDQ1LjQzNDZDMzYuMDY1MSA0NS40MzQ2IDM0LjY4OTQgNDUuMTc2NyAzMy4zOTM0IDQ0LjYzNzNMMjguMDM2NyA0Mi4yMzIxQzI3LjcyMzggNDIuMDgzMSAyNy40MjQ5IDQxLjkxNDEgMjcuMTMyMyA0MS43MzYzQzI3LjEwNzYgNDEuNzY1OSAyNy4wODY1IDQxLjc5OTMgMjcuMDYxNiA0MS44Mjc1QzI2LjQwOTkgNDIuNTYzNiAyNS43NjA0IDQzLjMwMTcgMjUuMTEwNCA0NC4wMzk0QzI1LjE0NDcgNDQuMjE0OCAyNS4xNjQyIDQ0LjQwNzEgMjUuMTY0MiA0NC42MjAzQzI1LjA5MjYgNDQuODc2MyA5LjY0OTA5IDYyLjI0NTcgOS42NDkwOSA2Mi4yNDU3QzkuMzY0MjcgNjIuNTk3NSA4LjkxNyA2Mi43NjkyIDguNDA0NTkgNjIuNzY5MkM3Ljc1NDk5IDYyLjc2OTIgNy4wMDA3IDYyLjQ5MzMgNi4zNDAxNSA2MS45NTg2QzUuMTU4NTUgNjEuMDAyMiA0LjcyNDI3IDU5LjU4IDUuMzcwMTcgNTguNzgyQzUuMzcwMTcgNTguNzgyIDIwLjAwMjggNDEuNzkzIDIwLjM4NDMgNDEuNDY2NUMyMC42MjUgNDEuMjYwNiAyMC45MDQyIDQxLjEzNTEgMjEuMjI1MyA0MS4xMTYzQzIxLjI0MzcgNDEuMDk4IDIxLjI2MjEgNDEuMDc5OCAyMS4yODA1IDQxLjA2MTVDMjEuMjc5OSA0MS4wNjE1IDIxLjI3OTQgNDEuMDYxNCAyMS4yNzg5IDQxLjA2MTRDMjEuMjU5NCA0MC45NTQ5IDIxLjM1NDUgNDAuOTE1NiAyMS40MDY0IDQwLjg1NTNDMjIuMTQ3MiAzOS45OTcxIDIyLjg5MDkgMzkuMTQxMiAyMy42MzM5IDM4LjI4NDlDMjAuMTgxMiAzMi45NjIzIDIwLjE1MTIgMjQuMTc2NyAyNC4xMjc3IDE1LjgzMTNDMjguMzc2MiA2LjkxNDk2IDM1Ljg5NSAxLjIzMDc1IDQyLjU0MDQgMS4yMzA3NVpNNDIuNTQwNCAwLjMxNjQ2N0MzNS4zNDMyIDAuMzE2NDY3IDI3LjYxMiA2LjM5MzM2IDIzLjMwMjMgMTUuNDM4QzE5LjQyODcgMjMuNTY3NSAxOS4xMzkyIDMyLjM5OTYgMjIuNTA0MSAzOC4xOTE3QzIxLjkxNjIgMzguODY5MiAyMS4zMjgzIDM5LjU0NjcgMjAuNzQyMiA0MC4yMjU3QzIwLjcyMDEgNDAuMjQ1NyAyMC42OTkyIDQwLjI2NTkgMjAuNjc5NCA0MC4yODYyQzIwLjM1OTEgNDAuMzc3MyAyMC4wNjEyIDQwLjUzOTcgMTkuNzg5OSA0MC43NzE4QzE5LjQxMTkgNDEuMDk1MiAxMC4xNTkyIDUxLjgyMDggNC42NzczOSA1OC4xODU0QzQuNjcxMzMgNTguMTkyNCA0LjY2NTM0IDU4LjE5OTUgNC42NTk0OSA1OC4yMDY4QzQuMTU3MzkgNTguODI3MSA0LjAyNDY4IDU5LjY5MTMgNC4yOTU0IDYwLjU3NzlDNC41MzEzMSA2MS4zNTA0IDUuMDUzMTggNjIuMDkzMSA1Ljc2NDg5IDYyLjY2OTJDNi41NjEyMSA2My4zMTM4IDcuNTIzMzMgNjMuNjgzNSA4LjQwNDU0IDYzLjY4MzVDOS4yMDYwNyA2My42ODM1IDkuODkzOTYgNjMuMzg0IDEwLjM0NDYgNjIuODM5NEMxMC41ODg1IDYyLjU2NSAxNC4zNDU5IDU4LjMzODMgMTguMDY0NSA1NC4xMzQ4QzIwLjMxMjQgNTEuNTkzOSAyMi4xMDY2IDQ5LjU1ODMgMjMuMzk3NSA0OC4wODQ2QzI1Ljk2NDIgNDUuMTU0NCAyNS45NzAzIDQ1LjEzMjYgMjYuMDQ0NyA0NC44NjYzQzI2LjA2NzEgNDQuNzg2MiAyNi4wNzg0IDQ0LjcwMzQgMjYuMDc4NCA0NC42MjAzQzI2LjA3ODQgNDQuNTI0IDI2LjA3NTIgNDQuNDI4NiAyNi4wNjg4IDQ0LjMzNDZMMjYuMTA3OSA0NC4yOTAyQzI2LjUxNTcgNDMuODI3MiAyNi45MjM2IDQzLjM2NDIgMjcuMzMyMSA0Mi45MDE3QzI3LjQzNzcgNDIuOTU3MSAyNy41NDA5IDQzLjAwODggMjcuNjQzNSA0My4wNTc2QzI3LjY0OTcgNDMuMDYwNSAyNy42NTU5IDQzLjA2MzQgMjcuNjYyMSA0My4wNjYyTDMzLjAxODkgNDUuNDcxNEMzMy4wMjY2IDQ1LjQ3NDggMzMuMDM0MyA0NS40NzgxIDMzLjA0MjEgNDUuNDgxNEMzNC40MjUgNDYuMDU3IDM1LjkxODIgNDYuMzQ4OSAzNy40ODA0IDQ2LjM0ODlDNDQuNjc3MyA0Ni4zNDg5IDUyLjQwODEgNDAuMjcyNSA1Ni43MTc0IDMxLjIyODZDNTkuMzM5OSAyNS43MjQ4IDYwLjM2MzEgMTkuODQ5NyA1OS41OTg1IDE0LjY4NTRDNTguODE0MSA5LjM4ODIyIDU2LjI0OTMgNS40NTQ1NyA1Mi4zNzYzIDMuNjA5MTVDNTIuMjExMiAzLjUzMDUyIDUyLjAzNzcgMy40NTQxNCA1MS44MzE0IDMuMzY5MjRMNDcuNDc4NiAxLjQwNjg3QzQ1Ljk1NjcgMC42ODMzNyA0NC4yOTUyIDAuMzE2NDY3IDQyLjU0MDQgMC4zMTY0NjdaIiBmaWxsPSIjMzc0ODc0Ii8+CiAgIDx0ZXh0IHhtbDpzcGFjZT0icHJlc2VydmUiIHg9IjMyIiB5PSI3NSIgZm9udC1zaXplPSI4IiB0ZXh0LWFuY2hvcj0ibWlkZGxlIiBmb250LWZhbWlseT0iQ2F2ZWF0IiBmb250LXN0eWxlPSJub3JtYWwiIGZvbnQtd2VpZ2h0PSI0MDAiIHN0eWxlPSJzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMTsgc3Ryb2tlLWRhc2hhcnJheTogbm9uZTsgc3Ryb2tlLWxpbmVjYXA6IGJ1dHQ7IHN0cm9rZS1kYXNob2Zmc2V0OiAwOyBzdHJva2UtbGluZWpvaW46IG1pdGVyOyBzdHJva2UtbWl0ZXJsaW1pdDogNDsgZmlsbDogcmdiKDAsMCwwKTsgZmlsbC1ydWxlOiBub256ZXJvOyBvcGFjaXR5OiAxOyB3aGl0ZS1zcGFjZTogcHJlOyI+Tm8gcmVjb3JkIGZvdW5kLjwvdGV4dD4KICAgPHRleHQgeG1sOnNwYWNlPSJwcmVzZXJ2ZSIgeD0iMzIiIHk9Ijg1IiBmb250LXNpemU9IjgiIHRleHQtYW5jaG9yPSJtaWRkbGUiIGZvbnQtZmFtaWx5PSJDYXZlYXQiIGZvbnQtc3R5bGU9Im5vcm1hbCIgZm9udC13ZWlnaHQ9IjQwMCIgc3R5bGU9InN0cm9rZTogbm9uZTsgc3Ryb2tlLXdpZHRoOiAxOyBzdHJva2UtZGFzaGFycmF5OiBub25lOyBzdHJva2UtbGluZWNhcDogYnV0dDsgc3Ryb2tlLWRhc2hvZmZzZXQ6IDA7IHN0cm9rZS1saW5lam9pbjogbWl0ZXI7IHN0cm9rZS1taXRlcmxpbWl0OiA0OyBmaWxsOiByZ2IoMCwwLDApOyBmaWxsLXJ1bGU6IG5vbnplcm87IG9wYWNpdHk6IDE7IHdoaXRlLXNwYWNlOiBwcmU7Ij5UcnkgdXNpbmcgZGlmZmVyZW50IGtleXdvcmRzLjwvdGV4dD4KICAgPHN0eWxlPgpAZm9udC1mYWNlIHsKICBmb250LWZhbWlseTogJ0NhdmVhdCc7CiAgZm9udC1zdHlsZTogbm9ybWFsOwogIHNyYzogdXJsKGRhdGE6Zm9udC90dGY7YmFzZTY0LGQwOUdNZ0FCQUFBQUFMNmNBQkFBQUFBQnVrUUFBTDQzQUFFQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFHb0VnRzRHRGJCeVhPQVpnUDFOVVFWUXVBSVVNRVFnS2hhdGtnK3NxQzRWQ0FBRTJBaVFEaW40RUlBV0RmZ2VPSXd3SFd5OVdVU09lZGdLLzJ3RWNmYlI0dTB0VXJ4bnU3bGFGNlVnT3BZS05xOHl3Y1dDQ2JiNUU4UCtmbEhUSUdNQWJBUGhmMHlxQ0JJSms3azZCck16Z0hkRkJ4MUdSeUVZNk9kd3gxeXdXeWJhcEhjTUs2dFhsQkpobmJBWjJ5bHVHZWtLRTVoSTNMWEZoR1lGOGZPdU4zbkVMRHhYNEp3R2hJSDY1Y3p3RDEyL2QvRy9TZFlyM0szbXptMTlTbGJyMEdPbVU0cFRNRk42c1IweHhIV0M3RGdpUm9obGtrWXVQeEQvdlgxU2QrM29BYnM4YWpnQkdJQ0lhRWY3d05MZC9kL2R1emRnYVJxMmdsYkdOMGJvbU9nU3BNa0RRUmlYRUtNb0tETkRHaW05a2ZZeitmbEhGcWJpSTR1LzgvOGNCNjl5WE5XbEFpVWJTQkJJVWFCVHNMT2lDTHVHS0xORWYrTDhaL0JvWkQwcUhRRTN5Vi9PbjYxaVRlZmI5V0xzUHZvZ3RZdHVZZVFsVnpRenhqblFLMGFVRkNpQWcvRWZtRmpLamRPcFZEUEdmY25zclY3aGFZQUtoSldFR2t2NGd1bEQzaEttY1dGWk9QZHg3M2oxY0tzWm5UUkQ3b3VTYjhwS255TFlkYzAyVVRBQVV3RDlVZEk4ZnVCNS9EajllWGprWk9SazVhZmxTRWltSnRNVHZOYlVTVXpHVk8rdU1uaTRMVVNtUS9xb0NHajFxelM0YUFBbk9VSHVpUk1xczk3d2JxR3l5MXJnbkdJTVJDSWtVcy9heHM4azFLVmFpWDlQUFdmcngzYVUwd2dKWUtGOVJpYjRuWmV5a3IrTTAyZ3FCK2I5cVdzV2YyU1BlM1A0U21ZRkZUVGtoMDRJcHVWYTUxLzA0VHVVODBSdWpWNjNaL09ib3EzMjFyNEFTU0FRbElVQUNHYXFSSlhXaGFybld0ZFZtUHFmK2YzWnJ0K25uWFAzTmJTcWFWTy9acW9FVzJzbU1ORmdNekJBa3NFR0FEYmIwbm1UTDZhMTkveDFwVURZR3h1Um5qaUs0d1JqdzBQYi9mMXF6dWZ2SUZyZnZoTlpDTDM3NEdXSDJvTEFJMnllWk9iTXZmMXUya2gxYUM2VjNEUTZocXNMTkxyWDRya3EzclNrd2pvUFFDT0dCU0U3ay93TjZNUHUwMUI0MDZWTlRteTArSUdnbUU1WHhQRFVscFg0anJ3c1VDajdFUEtGSUo2MmwrYmV6b3hTMFNZZEtxU2pJQlhXZUFCWkNENTZ2NWZ0SmRiWjFOMGE4dFdnSm0zM3JqTXlnMU9GZHovUy9hb2VrWkZHbkFFaDNqMERQUENjb0wwQU9pUnN1Z1o3Ly9kN1Y5M2FIbVlUNDhJdG1FaDQ1eTY2R3FMZVJ6TFM2UlhOSmI2Uy9oVVRLNHFHWjVrd21jYWhVazFEdzBvaUVTTVQvZnpQTjBtcGFVTVpGVURiS0pwUVVaTTRGa1RLRjd1UlY5LzBpdXQ2dkpydmZiMUxBTDJBRS9vSjB3Q3FPRHZnTDQzNWpQRWdaN3lQbmRyZUtJMU10MitDc2dYWjlwblNEM0VZYjVCc2wyb25tS0ZyamduQ1RlUE5vZWZpNE5ON2tqU2VvSVkxUWFYOVRHbTBXUTVsUXcxSUZZcnBBSjNhZzNlQ0EvT0R6Skw1OU02ZDFkUGtSRVZHN015dEhNampiSXhDVEdsYzF3Z2doVkczV0RPbjU4cXRkODBxUzN1ZDlxbXBGUkl3Ukk2SldheFpoR2hoakt4aVEyL3BmVEl0NElJaTNVK0R1aTJ6Mi8yUTcrY3FKdnMxaFVHcEpJMjNnUjA3N3VzL3RXUzIxaTRwSTRzMWZDSm9iWVdST0UvenBpZ0pGZDVCVHFvZ05RWmFtZmc2NWJNUWlqanlXTzN2d1lDOCtYS2FXNnpYeVpNM2NZZzdQTTU4WFdzSXJyZUllYTNpZGRZVWJiRzQwUVFtREpUM3BNQ2tNVEp3MEhFb1JBQ05KdjFkZ0tBZGgwRFhYRG9rQjlBTjNxNW9JNkNCQTE3VTNzbmtIN0ZVeEVmd05aQ29RZkUwSTFZQkQ4RTZFWEIxb3ZUQUZTQWdZRUJTUUVGOE9vVUtEazdKbWhsVkZTRXBydTZuYW10NEZmYVloZ3hlVFFQdXRMUCtodG9VRzNSdjcyNU1nN2NKNVFsc2ZtbXlIb00zOWo3NDUxV2s4Um1Ld3VqdzRvdjZ0MTlRLzZQQmJEWUFmNUN2d2lmcTlyUS9UNm8zT3YvS004Qmdld0YyNEV6Y2IxN1pxRDdWQ3VzdWx6MjVPclUwQmRhZ3dpb01iR29SN1Y4YkwyMWVNYkN5c2k5WDFpcVBtTDEyeWNnSmJXdFpsWnJQYTltZ2dHQ0IvdjF5K0lIOVZsLyt3VjN1eWUwRGUyYVdkM1hIaEVQWmdoNzBaeUkyNEdwZFducitZRnE2OHptbUpMOW1MVzZFMWFMTGJSamU4aUZWZTUyd2lNV3ZQeE5IT0VUTzZpb0VXTFRXOVpPZUJVNW51bXRyZTRSbU1RYWZ3MlhweUVqWFhqVDNkdVdVNDF4RFNwaVVUTkRnSVpLVlFOb3prN2tsdEgvbVVSbldMblZseVdnc3ZpMDU2T2RkMnZwdjBTcS9OM1hTUWoxNDJ4TFlGUWhiM0dzNTQ4T1FYMExyR0pjWjZGZG5DSjA1Nms0cFpIVG5iT0h5am1QTUlBQmtXVnRneEZsZmtlT09IbGlCQ2lTT0J0TTkwQk1EYzVkR3ZYUWdCenBUb1pHdHMxUDBOSVVFSlRraENFNVowS0JJa2VldmkveWY3dkJiNURIZy9lSE84dWoxalBhNTRkS2ttVk4vZEtjclJpdmtLaW55Wlc3dWJpMnRhcnhpN1k0emFaWTV6anRNRko2SERLdnZ6OWs1MnUreU10cWVsNHlSMUVpZUpVSHhCdkV1MFE3aExtQ1hVMjl5eEhyRk9zaVpacFZtTVdod1ZWQW5zekZmd3ovTDF2SEhjODl3R3puSFdEdVlLWmdkekhqT0w4WXhCTUoxbEdrZC9aWExGWkE4dGlkSkVWcEZwcEhGRU5lRWsvSUk5d0M1aFc3RWVyQU9iaDJWaENaZzNKc01FNkQzMEFyb0hYWWEyb2pWb0JocUYrcUpPS0ExNWd6eEN0aUx6a0FwNEtIUElKSDVUSVV4Q1lVeWZRUng4T0V2V3lJUkVhQ1JQMFRuL2FHb01Od3dxWHJoOHBES1hyb3l2T3ZYWUdxa0lrNTlGbW1hNktKMksvWUptaFVVc3VINXF5UTF0WGN2dVBIeHkvNU5uNno1ODIvVEgyQlYzSCt3bE12akxkTjhRU3BDN0ZVK3FtclJuQnNhZGZEcHhhTzR0b0V2UGNsRytJNDI1enRCcFZsMU0yNURYU2NDdjZWdkdEZGdyZlhYUUlYSklUTnRqeHpRMi9odXJuckdscVJIZHZ6MFh4V3A4NmJXbk16STI2eW9qODdDbHY2eDZ2NDMxcjdtS0dwKytBY05HZWxyZXA2MEswSWsrUDczTld2aGsrRHVsS1VOTFU4L3AwNGN6aGRNWjlUSXcrcnNiZU5xNndNUmNiVlhmL3BxbS91dUdldEQ2Wk1Lbll6L3pJK0FROG9rZWZqVXdIeDNOVXNVYjFnRTRsRldkL2V0bEx3dExHcHZkd05Oa1JEVnI2cCs1M0FBbFVKeGpYQlVZUVZUUWNBaUl5Q2lvb2QzL2szR0ZFU215bmpvYUxpMlpuU1drUzZnU1hvbVRtSFFuOHo3WmtwTmNVa3JLb2dwVm0zMDFGTS9qMjdWd01zTTVJOUdvTWVNM2s3dG9PZXQxTnVZMnMyMjcyY04rRG5DWVkzSWQzVjZQODRUbnZGWi9MdVFPc0lHcU9UTmV2T1B6eGpjS0JnVUxZdVl6WkVjWkdHeDM0azBVWXppV25qSENuR0tGSGNOekJDSWtJb2xVa1hVVVVibWEvWUNuQy9tYU1NUG51eVF5VVRHeC9qbitKcmd3U0w2UmtsVHZqaVFySjd1dXJtUnNaS1NoVDlzWDJ0T0pydlJrdmVsVGpOWHVVRGU1WkdnZWZvM0lZRDVheUdKdktjdlp5clY2MTZ2ZHViYmwzcHlkZWZjWWtjRmxkSlhyM2sxdXM3dnIvajVXdTZOM0w2N2V5djhCNWxZRE9UblhGelFJdlJUMUVpeWdoVEg5bWhWbnRuUUxSZFN6Uy9jdTlqM1NhSWxKSmVsdUdRYTNSTDNUSTlVYnA5TFNpNkFydW9JdUdhWmhCWGFqSkZaRWRlb2VvODZCeG85VDhhcDR3V0xLUlN5aFMxVlJuWWxlSTdWZTVoRy9xT3BrVmQ5UHJPUXJ2dGwvcVB3ZEF3anpwTnBNY0NjTDlCYml2eE1ENGhuZlpxYjFxdG53WnZ0ODZMM3p0Um1EUjFWVnZvUlppVm50MmpycU9odFVwN3A4ODJwRzVNN3U4OGdBV0o2WGxXd0t2cFl6RlQrbGtXTUxWcHUvNVBydWdOdkVyV2V3TTQ1Yy9Gd3UyVUM0dWZVVUh2UUp2S0RaOUdndEdzWnBkK0hLWUU2Nksvdm02TXA5ejJOMUlSUHN1c1k2eHJIUGpvR0ZsMnRkNnpxZWNseGpyVVZaTHk5cmpYRWM0OW9TYTRVenhiTUEwSnV1VlZ6RDFadWRCYWlPems4SUlDSWZ1ZG1PMzhuTW04M0xqSWg0a282bzZ3cTBna2U4WExPZFhGVEFKY21MeVdLaEN6d3RQTXl0TmFhSjFoWjAyNm5ja29HQkdNbmYyci9xWGc4VUZWTk5yTldiS2RwMElreS9GV3l0L2RxMVpQUzQrV1pxV0FaN3lyNHFRR2VPY2p4eDR0UmxyZVZuRjhNSGI1QTlIS1laaGdBOXJlY2V4Q0RQb2JvNUc4ejh4RjJsTEYyQjJ1SXd6c3d0ZTIxR3ZMajBLK1NGZUZoWjdtN2ozNDBwUHduVmNZVitQV2tsS0xsOHpLMnZ0Ym9tbVExLyttanhsZHFpSHBYMWFsMjErNE5qOU5YTkxnY1kvNWVRUlRCTjNBbFVvY1IzUGY5YVYyL0NrblhpRWtYMHhXUG4zR3VLVmhhVVU5Z3Ntb2h1SWp6eEpDRTZmVWtFUjNoVVRJanlvSEc2UThJZWxVK21Ub0VwdnU0SitQT0pjTTlkeFJOWUhhUmdqa0FsbzYrbngxY2pZRVFtTWFUR2dJeGVicmRPUlZVWVFESDhsZEpZSkdRWGlIT0pYbHBXaUl5dzEyUUVCZ2lFSHZDdmdVZFlpbjBGVEU1OHVpTmlIZksvWU8vVlpmWTZQdDRIRzdrZWJBWUhQaDZTRTdwRFRTNmtOSyszb2hPaklzbnA5UjFuVnZaWkZTczdXYlcrWUU5aVFaL1pLUklVMzVyME9TTjdhcm13blJOUFhTRkNDNmYwWjlPVGZVQkU5SmZzbk5NM3hRSjl6QnNkbTl2dDFrR1JlbHE0ZFFXY0tjYkJHUUdVallLbFhyTmx1dFhxVWJmWmtPaG5KRmo3UmhGOVBxMXc0V3NoMXRHTk43R0w5S1dVb1lsTmFaMXpxUklxR3FnU01SZFQ5cThueDdTQ0tNS3RTNGtsNTJwekcwMXVzZlR5Q1hJUUNSSU1iUzZjMWhySkFmZWdiNlJTUE84bElFRlh2VmNDUm9RSnJVcGh5OGVOcSs4TU1jOWxtdG9jN21OREl6b2VQeDFlU2NMYUoyamIxQjdibVdmdGluNXVqclRFQm9kSUVVMTFjbzYyT0tUNzlPeElGaTg0ejBSbXh6bDBSWGVwNkNtREFvcDR1djA4R3JyT1o1SHo3VVVJa0lxZU9DaWdlS2F2cUpZbGNkWmcxUFFCcHVwZENJZ2lHdDA5MTY2NDZWV3lnR1BWbnFaYmhSQ2szMzRLTXZMMnFNV3JFZFl3d2hCRExmY0tHY1dDMGdwTWY5ZE82cVBmd3Y4ZndnQzZjOWFrQlUxeG92MXhiRU9SK2x1aDhBNlBPMlBxcFc0V3ozMFJGQ1hPUVBQaGNRUzRFbmVoUXF0Y0tXS2xjQWovd2dpVDFReUl2aEdEbTFtVDBVYThTQVZhYk1GQnFMUm9BUkRzeVcyZVEvenNMNDlac3MyWFVodS9KTnhBZzZ0NnkxL2VWa1VVcUV2WncwU3ZpVmluZExsZ1RFVmpQRFlVU2xRODN3VzV0WjRHTlJVU0JBYTgwYkRRUGZuUXpyWGw4WVVOSWRreFBLNWlJc2t0RkJtSFZ5MlJ5Wi9GRCszRFkxQS93dVhXQnJaTTlIdHlDeGlKUkFUSmFOc3lUUVlZaVVRQUNRSWNBaGlKUklScEFKekFPLzNZRURTbWp0VzUzbEs2OElJcnlvaXFGNE50VWxrcWNQZVRSZGZQN2duR3JIaHFOcUFMVXZla0FpMDB1d0RoSWxFZDZiWXhoSzRzUG1PZWQzNm01bTZCWExuQ1doc0tCZ3R2M2VMNWg5cWF4VTVIVkhzakpEMUN1MzJKRFZqSjIrdm1ra0kxcHNzYnFwUlNHUHNyc0ZHSGkwYlVLTUpvZ3pIZ1NVK1ZqV0FFakFLNEhya0lrRlhCSlUrMGR4UXFwdHNZOU1CRkhNdmkxZ3NNUkNEOGNQWitLNnpUWHE3dUpOLzJGMjJ1cHBOVm5oWmE2ZnhnNnExSjk2Tk1YTTlTMzNoUFd5SlZybk5ONzNuRlZ0VWsxdlhZZmhQc3g2dThQRjRleDNWc1AvVGljVnpYMHlPNC9KVGplbms4OExqOXFCREhjWTE3czU4NHVvbW90SnFVcnNwZ2dNQ21yZjZBWDhHQ3NzZVV3ekYvd3JJSlNKckRhVW53cUtpRWdhb2Z2MkNRbWNkYnRXc1crTURGRE8yV1NpWEx6ckNld0Y2RDFWU0hrUFcwVFpHeVFvSWpUeWVTRWt2WU9mTVFIRmd2eWY4dHBpSnREc3RhdUM2VlZsS0M0WGFsZW44NHozSG9qNmZhci8wRlFkdldCZ0oyNlZ4STQ1NENSOEJCMnRhNjJGTnFzd0FvSWs2NFpBeVFIaU5SSzFBZ2pOTWN5alVNQTlYMkR2SEpmV3FQd0RtTDdHK0pJY2xjdk9LZlNLZ2dNR0FWZnNDQzkwa0JoRXVIQ0ZJM2FlYjdXbS9DRXgxYk1IbU95QXRCR0s5YWdiMlo3cDZuSngwSmhEZUhPb3pVT2QyeFBVRms5aVRQc29Ba3NTTWdXWXZVNFBKNjY3WlhRZ05LUy9OcjExNmMyUEJLVG9NYklBY2psMmlSMHNTRDdoQlNaa2pyYitVendTUHNZQm9LTGhjZXQvc0JTd2o5VWV6RW10YVJwVGJxN0xXUUMzRDdOamFoZ0VjWU03T201NmpqWm1WRW5TeGdjbG12bTB1VEZxcm1NcHRWbkQwTGJYT0xzaGg4S01ZMnBhMmtyU3cxd21rd1VIWXQySks4Z2tKY3l4Rk05UHBmaXBRQzNZeVU0eTNiWEhGZ3BWK2s5T2pnYVpWTnY5MnJXOWdkejZBb1Y5K0tsN0tIcm5nbGg5Z0ZGRk1oNS9WdXdPSjdsdXFRY05GV3M2dXRxeEFsTktuVWFwUlNLcVhiVG92UnZwQlNLcVZVU2ltbFVycnRIQm50QzZucS8yU1VTaW1sMUJXSVN2KzE0bmVieHRuYWlpNTFGVExoRmxGUmxtSTZpQ2dJM0pUQXh0SUtiQzNjYzFSendGdFYrSWRvZGpUc3NRaGJhYTFTOXEwRUw3alcxQTRUNktLbXBiMVN4cmlLQTFTdTFMZmpxUFo0bUE4N0F6Lzl4a2FXUFRqWU5HTUYwNDFCbW9KREptcHRnYlRJNVQwWUNSK2FtVTJibVgvVXRNejE2bmNPTEpLdVBlN2hyLzk1cWRaaGcxSmYwUWcwVGFwdDE1SldVOFBBZ0xTS3FBQ1RIdk9wd05UMFFrNGJOLzJlckROLzB1bEtpOERCN0ZYenoyTzUvM1R2UzRBelhzSnp3Z3dQaHQrNW9oNk12eDV6bnpFdDBtaVFVdzNwd2lXTjJOdERUV1IrSmIzV2JXWjZobUVZWUJpV21RNnlySVRMTUF5ejNaRTNtZ3dqc2dTNE1qSms3eTVzR0JuVFdCYytsM2RuTjIrWk5FTzdMV3drc25lc1ZDQWRKeFlkUG9vaWU2a2Zwb3pOV2V2dWpBci9aVWhydWF3STZXWlBGNDlrUU9TOEF5N3Zja2tTc25OWWpERzVKOEVXOUl3cDcyb2grRy9BbFlDSFgyUmhwb0hXOWNjMXBUVVhjVldJa0NtUGZsazQycTdQdEsrTWUwa2JDSE5pMWhlaFA4cDY4enlRbWMyV01qUzRNUGEwVDJXSzFtTmpLbHlTNUpSWFFMOVJzRlViaHhERXFTQXJLRC8wN0FYWmpCd0JleWlOUTRpL0FDbU40VDVDa2xManZtWkUwL0NXT1g3Zmo2bmMxSmkyaWI5cUtITGRod2ZNb2N1Z0M0V3A5elI0T1h0WjhSUGR5ODFDbEw0N2c4cVY2R1htY3piTitQWWVzOHhhaTRxaVVGTEI2NERrZGlFbFcrS3JvaXE4WmdSUEVIL1BKWHRxVW04emV4dy83ZG02eGx0ZlB1VG55TkNQdDMyVzY1Z2JycnQvN25NZVZIdHBmOVNLbDdIaU9HV05Rdzl5V0xrU3VReGhaY1pIak5xeHRCcjZxbDV5cUlUY2hKUnpQWGpNT29yY0QyTEdWZFFkYy81QXh5NXcyNDB4bW50WkJNZFAxOE1rWnY4dWpBMytMK0ozcUI4UWYzUmd6dXJ1emhpenZnVEkyM1REeVdWVFprRU82RnhuRlFPT0Y0eGJhbnd5STFiZVdJOHh3M01rSWpHdDhJQ0JLRWdTNjhvWGNQOEx0VDBhdE9LZ3BLTHRuSW96T0c0VDFFaTcrdWl2Wk9KaGNnK2l4UDRTVHZndVZFKzd1bDVJNi8zMTMxc3o0SHk2aEdwMXJETFo3d0x3clRPQTg3TnVlR21UTEZGL0E4cEx4ay8xZ1BFd0tKT3lDQkhSWStpbUl4T044SHBkbi9zNHZMQTRoeU0zVDU5b2kwL2xyZW5VV3FlYzF6c1pYU1h6cjZaUzE4NXd5T29jMlZ2bTdLRzdTUkhCMVZxOThDc0FiYzFxSnd6TkwyUDlrV0VFTmtZbEFZTGhsWlNRYnMvTStMSjg2Qlo2a0F6R3d4NDhJa1NMdE5WWUFlTXJCMWp6NkpPOHZRWU12TkN6RU8wMUFoejNLVFo5QUNJcEdDRFQzQ1QwTitqZmE2aGtpMkVCejNaNnhxakFjdXV2a05hVTBpRjFnKzhmMUg0OWNNM3BmVm8vK3NGRVB6OFVHcTlFNXJzTmVINnl0MUxqTDVmL1hPOEV6NmJTdXRabXJpYXBsbWdTbWZHSXF5Z0lGSDNhUVMxS1RPRGtWL2ZRK0g3TjljV3I3RmlKQWNKc3c1QmxFT2JlWXQvTDFFbWhjYWEyRkhBU1NsTUJucEFCSHN6Mnhwa1FZbm5zbUpTOVlGSGMyRDZZUGN1Znc1WlgyMUVTT21Pb0RHdzFuS0hncEZNTUxkR01UWWhDemsrVlE3dTZwMjRMWk9RSG9lMHJjS3lBdDlOa09saWgyU0xpRkdxbkVCZHRhSU5mSHdjQ0J3QjhiWEl6UlhOeGw4SDZEeTNrU04xbS83QU9DVnh3Sm1XQ3FhSVN2TlV3bnB3WGVGZ2xCdkRZejZmb24xU3VnWTdHeWs3eFJUdUNUNzgyRGU4c2VzWlJoYTVwWHExbXJSSndUaC9hZHZGbzBFVUluMWVtL2hLREV0dkdYci84SkZXcVFtU1BoeDNDRm1ESVVxd0Jla3c3VTNvTGNmbzI0RTdadWFhNWdMMUlkQWhRZ1dBcG1uVmQxRmdQWVpxYmZIVDFXU1dvL3h4RU9JNGZCQW8zb0xlK1ZsVVg0MnQ3bWhWVkhMWm1xNkpRcloxNG8rVGJQcTlDUisvWlVMZVlTbm04b1VUMEpjbFhmd1pSYnRITllUWC9GUlZwRThQUHJ1QVd4TXQzdVZPdnRWbjJHYjVVWTFKdnZZVE5pQVFnVGFsU2ZFNUpXb2tOV1JmUSs3MlFodEJocVZRQzRaeFRWd3MrYVdlSGJ1WkRCSDV6WGxsL0l0REpzdjloSzJhbHR5RXFLbTVkQ1FkejMvZ2Q4RHBXZTcxU0RkVG9pMGN6WHNIaUszcUxLbVJad3Q5cWRJTUpwd2pYM0thSU5PZXJFVVZhOUpEb1NnT1BTbGxTNW1VaFI0OHBCak14emxnbms3YlZlQmxNNEh2WUhQUExJaEhsYjdIRk5Kc3dKUHZTbDRMM0xDeGdVcHVoeDh0MVJiY1NqK2wyWGtoZWJiL1ppUitPNzdtalZrSFN5Z2V4K3kxTHlCcXJ4YWowaXgrZTcvVlhRZngzdU1kdXhWWEE1ZitDY1BuRjV4WG1ad0xoWjJsMk1SQ0EwUDF1ek9VUS95dnI4VmROV3pXSDhzZWVPY3VEM1ZLSC8wTVZYSkJ4SEZoaVFFQi90VEI4ck1uL3Y0dWlJL2k5UFgzdkQ5SDkvUHgzSURFQkZvcnZqZnUzZmhEc1YvY0wvLzBFOThaYm5uNHdLc2lDenY4ZHNvRUd4UmNpM0h3SVB5dFNGTHR5UGQ4Tk1UMlBOc09zQ1ZuUXpBTFk3am1qS29PZThyV2xodnhiVWM4b3hZbVhJRkdTWkNsU3BVbVhJVk9XYkRseTVjbFhvRkNSQ1lxVm1LalVLYWVkY2RZNTUvM2pnb3N1dWV5S3E2NjU3b2FiYnJudGpydnV1ZStCaHg0Tng4UjN5WmVmY2NaVDA5RFMwVE13OGhjZ1VKQmdJVUtGQ1JjaFVwUm9NV0p0c2RVMjJ3M2FZYWRkZGhzeWJJKzk5dG52Z0lNT09XekVFVWNkYzl3Skp5R1V0eVNoYnBqYVhEeDhzVG9EQzRsSU9CdGp1NUp4cS9mZ3VZNGdzZnk2bkZneWJVV3RwcHNMdDFrd3I4TXlLMUlDN3JYV1g1KzgvTEdrUkVieUpoR0lScUtjYUVra0VENFFiaEFPRVBvSU13aEZoQ0NDRXg2Q2UrSmluQWEvd0h2d0tGd1A1OEV5S0xQbDdiOVROWEdqb2Q5NjBNazIxMUZOSlpWM0NhTE4ySmdGQlc5Y3NjY3EwK1RCRUl3V0srVWtKQyt1aEI1QktidUdRV2N4ejJMNFJOd2VDSnAvekswMHJHUVZCSFZoZG1Gb1BkUUtRVW5IYktBaGZqRUVaV0ZtWWZCNGpKZ2dwMk1HMFNCN25IRXBUU1VHUmlnSC95L1RtZ1l3SjRwSmtkN1NRaDgxV0Z3UW9qZ3ZNTXNhQVdUYkdkVVlUVnRXQUlaN3U0MzhqTWZkb1k0WlN3ZXlFQ3o4VmlKUGFDdGVTenRKbG1yTXZsbzJtVnlGM0UxUXhqc0VXcjNpZ0xTa0l6MEZVRENGVWhpRlV4UmlHbm9oSWZ2WTY0Uk5xVmtRY25XNlRZVE5mYW5wQlltZ21TeXFzME9ZNHBOdzh4Njc0bm1GODg3bk1jbDVPN3FkdWhIZ05BazU2aHlZcVVPZ1VxbndUaVVFTFpZMnlRb3l3ZHhOK0lGUkVzYTRNZVRMT0RVVDZKa29pRWtpbUN5T0taS1lLbzFwY3RoRUVkT1ZhVk9uaVVFc1loT0h6TWlDck1pYWJFaE1VbkxJeTg0ekFuV0RTSUF3SWhCMS93ejNTN2t6SzVuVGVjdWN6dHNLQjg3SDh3NlhuQytoeTNrWGxqa3RRNFpURXdyVUJaQ3BJK0N0OEM0SXFxUmd4RElPdjJpUkNVcTNXbUhDMDdTNi9sT01qbmVlRXBGazlVaEZXcytiWmRscWU3akRMdjNwRDltbjBBRWpKanJxcEVxbm5WWG52QXNhWFhMTkZEZmMwZUtlRjJiTmhkR3p4YURkOWpzU2xwbk0wVHIwQXJiRzh3UW5uVStpeG5rakZqb3RoTmFwQm1ucU5MRFVSc2hVTXJ4UUNVQ0lKUngreXdDZ3pMM3lIRzY2anJqbW1VaTkwMXUyQkk0MnlERjJjNno5SE9jSXh6dkZDYzV4b291YzVEb251M3ZtVXp6MlF1cEVjS05EZ0RBaUVIV0h3THVKcEhJNmI1blRlY3Q3STQxYm5tQXNPOEs3WUVqMzRnMTdQRS9jNWhsUDk5aWU1aWtzMCtiSlprK0hRSzROczJrTDRVQ2RWUUpnZzhlSnN6TzVVWWZpRThBK1pldlhIOHNhYjFFR01BTE0vb3lOUEFEYUsvTUJYQVNRczhzQzlCVW5CWmVDMit2ZzA2R0g3MU5RaFFEdkRxc1R3NG4xa0pZRzI2VHhzTzdHWWRVQlZpUUVXVWIvMmJueU1zWGZoTGdsdWc1MHNqdmRPd1NER0FtVDlEbENnZEJhS0JiYUNSVkNiNkZldUZrRUY3UEUzTDhkbHQ2RlpMejE1RVZaQUd1ZDJVSXpvV1VSdWRCclBINUU2UHQvZmluTXY5V3QvNzlyNTdUbVlCR0w0Z0dmUHdIdzJmLy92K2YzZWZUaS84VC9LWTltUGJSNWFML0xzcXJmMHIwdmVjL2hyUTlWUTNuV2grM0JYL2IyaVpEQlkrNXpYRHFjL0tBc2ppQUZXZ2dWVDJYM1ZIeTM0Sjk2VGdSMi82ZEpOK0wvazBiRGM2OE9TUmlQN2RLMU1UbEtlVG51MmFKMmtnMzQ0bXQrK1Bnb2ZwKzltdmpsdTk5Um9nRUNGRVNDSXlLallXTmdZckZneHB3QW5aZzlLVnVPN0pKejRFNU9RY21iaTlXcWxLdFFvMUsxV2cyYVRURlZremxtbW1XMlNkcDE2OVJsaVVVcExMWkduMzRETmxwdUV5ZExNNDlaeEpIa0dUU2o2MFo1b1hmOUhqVWV5ODhIVnpyb0c0QWpmd0NvVjRDYTdqRGlBR2czbzZBOGZqNVZvYzM5MUEwais1TnlmYWRsRDI5aHNFUG9ndmFFY3lLdEZRdTZkU25hT29RcVBLSzMxWGpWaysyblhvMTdxR0RxVVh2dUllaUtOVHBZUmF2MG1HbzhqV2wwYU5NN1FkY1NTSWFXdDMyYXRvamtnaFZDN0haNFZ1VnJQNis3MGhzZEQ2VmhoZ1MxVGJXUGVCQXlwa09IZGlLcDBLMkFub0Z2OWJBZnNSWTI5bGZZeDRyaUZtMi9UNDJJVVYxWk9YcVIwNFN1QXFsajZJNnJ5VGRHbE10ekZWL1o4YzJxUFp5eTAxc0NOVXFyMkF3THJiVW94ZCt4ZWsxa1IwN1o1T3oxa3dKOXFvSjg4NTJySzdQbXoraWNXWmNobWV5UTZPODNzYlp4ZWx1Q01Tais4dlJZLzNSK2hIcXBwRHlqeWJpOFBMclpnK25La2pNbWhIcXdmRmptYm93V1BTdFh5cGI2MHJMOEp5VFlNNFZzZXdlbUd5T1pFY0V3ODZMNzV1NU4wY25TWTMyRVVTcnpRRmFLZ1cza3hzUllacG14QWswOWVSTGY1WEV1bHNrOE43YXR0eWJDMzJ1U2pVb2xEelR5RFdKOWpJRTFJWjBmYmNWVTB4dlk4Y2s2ZTVhV0ZXanVHMS9sWXBnWWxxQmFiZDFuTFdDdnRvNXZ0V0MxeEIveVk3cVplYmtiaXhOVGsrSnRLNk10TGJ3dDB1cFR3a0tTNjMvOXBvVFVCMDJlR0pLaUtqbmFIQ05LVFBsQ25BaktWZGVIc3hrMG1raHNOTjJmMTNDaE4xbzByZFVmYWRmZ0VLdWNpZVZkQVdFMlZtWm1GbTV1TXBtWmJiRFJTU0tyQVAwQ1RuUmVwcFhTeXRyYXltb1haYlBOemVlazYxUWFHSmEycnVuQkRIdUdmVmluUlJxZnNWRHJxSFB3MFkxek1KdXFkTFNpaDlzSjdLeFpiRGJiMG9KT3RXZUxMVmtzanBtY0UxbU5wblgwN09TY0NjbDFqVGtKamJTRVBVeDBuTm44eFdYZEJtUFIzTVZ5bVYxYVhYTnk4U0Z4ZUNYMkJpRE1RSmtNV0ZBM2Y3Sm50NWJsdGRwQ0t3SE1ZbVkvWHlFMU5OKzZoOFFRSXZhWExZa0JjTHpTdk5hbEora2QrNVRNaS8xOERHUjNIVzUrSG9YMWxhMWFFM0VXSmNXUWNqcVY5Z2ZCTytZZXpja3VjV2o1c05mbjBsU0JsNFhTaGRvOUxMaVlHTlpoeWhQTmJWK1hCT3JKUkVFUWhIUzM1VGsvZUwxUVF6VHZRcCtGbEgzUnMzNVNHRzJ4S0hIS0JCdkJKcHIrNTBuMi8wQTdwY1FtV0RRUXdxSVlnbVZwaVNUYSttS0VLS0pGbGNZM2k1S21OSHNzcmovMEZ1M3h3OE4xeUw0dWtNNXVubGNEWDVoVDdOUWlvaFFESmlTa3pzaGhrQ3I4ejQ3NjlTcUQ3Z01nWFg4Sk04OFFBQ1laSWZSNWtNem1LK3JPUkpYb1prS1dncVU5Q1FBVTVPT2piWlZiVkxFa1IzNDN5NU9WTk9oSzNLakx6Y3JpVzJVMzV3WG5xaDBoNFE2SmxsTVpCQXNJa3RHZUFSRUVtbUJ2YkI5M245QXlpQ29UcUNnQ2FFelBscGxrVVpBdmJJY1lFaS9YdFR1VHQ0OWN3YUw1ZUZHR1dEMTNqeHduS1hlbkEwUjFCWnJucVpyd2RzK1pBQnB2d01KY2pPYjdFbjhhNjY1b25mOFkxeEFUUnhkRW5YMmdqTGZjQi9kMm1DY2RCQ0J3OU5YTVdrRG1EQm9CTkR6bGVIRWVRQ0thaytiU1VJTzdLZjJVSEh5a2lDRGxvZWVObWoyejlnVjFaYmNjV01TQzExV2QzbS9mSDFxM3pKKzhDZUR3LzRyOHNEcGYrRjAwZDlGT2lDSkFLUnF3NUl1d2RXVFVVTFFMV1EwaHNSZjBTNms1ZVNuSmtVVmhGMnNmeFNaS1hzYVJhRXdNVHE2TUY0Q0l6bm9aaGZUbGNSbStqQUhGcHJEb01DVG1GN2RlZ1IvUnZJU3ROV1FDeW1vMnk5Und4RHhDcDlGODdtSWd6QjRHbXplSWJodlNyb0d1WHdFZ0VDU24yNEpGa1dJQTNxa1R3N1Axc3ZPVldNWkdCUHhXMnoyRWZPWEtYRTlTUy82UFVNMG15UDZTYWtkTW0vSDh4ajFtczRORmN0Rmo0WkxaVnlWOTdoSUsvL29ReGNRdVRCUmh6NTd0NTM1VlBxWnFxN3ZCVlZPVVFIUzNvWXowbnZWcnpYL3V5eHRqRW1wb005RUE2cTRZTFh0ZElSY1F2RE1XamlkdEdKNFFtMjJORDJqL3RpOVVTR2FmOW5RUGwxdUIzNTFFUzRiY2lEaE1kQWFiWmtqS3BRbS9LbHVpWEs3TE15WGl6SkNvM3Y5M3QzTjdCYkwxTkI4WWl2ZG1HZnF3bDhYeWxFUFNFTzNXZi9laFhrQ1pvVmcySDZIZnlwUCtzbWY0blAyQUZnRUh4SDc0OG9BQ0ZTZkJkdm8xSVhNTFBzc1FCTTlNTzdPa3duVlhwNW4rNlF2V0wyaThPTFZ2K0lJa251cjdzemRkNi9sY0NMeEkrcWNTSTlISURmWUxVeE9rdHQ5R29RYWcrMXNBeHRTQWZyd1J3VFAwSEExQjR0Wlh6cWtBalpDaHBwOFZDM0lTaU1NQTRhSlkycnNIRXEwVXk3WlpiVFRnWW84SjVwZG5tZm9jUHUzbHpzMWJUVlJGOXNZUmQ0ZElOK1dUMzl6ZkI4UDFqZG10NlY2dzJJY0ZteWU2K3RaSitRaVVuRFJ1SW9NZ2FhUSsycng3TFRDTEh3UEM2TTlIZUh6MXQ1Qk1BeC9qUVZ6KzU0d2lDSlNzZzNaYkp1c3dqaHU2TE5BVU1iMkwvdUxKcXVBTEtJQytzZ0RnbnpWcWVQSUFCWWFtVmJnYlJlMHBRMlcxMjUvdUtPUEJYVGRZNUlQOUFnQTFJbUpSUmQ0SUtPUjhhRktEcVNkNDE2ektkRDgvMkZaWElrS3NRZHRDTzYyaGhZeHhHMUtLQVkxTHgrWGNQY0pWYXluSlFXS2FNOXhrWTdyd2xSalhyWXB1TU9GTHRmR3g1NlA5UHV6WlBpU0U2aVlzL3lYb1oyV2E3K3lRRVhPM3RMMVNWQWNlMDlldTNUZDJPRnk5MWpIS2w4aGhpb2Q0ZWp4bWx3ZkFIODRZSnV4Wnoza0xxZlBubWZuTjAzT25CNVJPMmVTbXJ3ajNGOGJ0RmdTdjQ1dk03Nko5VVFZVytRczdVWi8rb0tXakYyM3ozaDdnOHBZT2t5M0Q1VnBiVGVmNUtMcTdzRTFaMmdEanhjVkViVnhTMlM1NnAwNGN4NW50cjZrckQ1Yys3bk1tcXNZUDIxYnZUUDRYUFZlZmlIL2JDREtLckNqdFVvK2VzNnoxcHNVYTZIL05QK2pCanpvTTd0ajJ0Q2lJLzNZbzNlK2wxeUVwaTExUjFyMEJUMWY1UUZncDNXTWFiNDlwWHhvcUxOM3lodVBRMXhUbTFrMVRJVDYwaFJGUGJ2ZmIvSDRaZW1SclkxenNKeGptei9WQVpDTWtKU1FRTFlGWU1ncEdiT2wrVGI1MmY3OWw5dWZZWGdqYmYzSFk2Z2lNaTRxM0ZtMnZ0WVVsQ0JVL0xJdysyK1lvSUxuMDBBdlVnbUE2cUVZalJPL2pVUnlzNkpmUGVoYWd3anRhaGxCWEphREMydThlaXRrTFFQNTJ1d1V2ZzFpU0JndnloM0krNzlJcFdseHJ0MDBFMldZd2Q3OTFQOVJMN2pDZWRuUzl1SUprWlZXZ0pWMS9FMlErbTZEcUVSSVIwNklFYXJqNUs2UisvUFUwdEo3VUtpK2lxcm1TL1VwZDd0cDNaUlN6Sy9nWXk5MFIvL2ZBWXozdm1oeWo5djN1SDFUVlVGN2VPVCtiZU1ieFArYTEyZEJ1NlMvRlJNYmlzakNXbXo4K1lybFVYSEIvNXNreE1TVk82UUhaV0l1bHBWNjZYaTZLMllCUlYxMUlZMlNJcXIwV2E4bGtZN0tzRlEwODVxL2JuOWI3OHZVM1Rldlh4L3R5ZXBiaDhaY3pVMGRTdkRrOG5TOVdQaGwrcC9NbE1mdGlLMHpTUDVRM1d1elJKRVRaRVY4OS9XSU5uOW56S28xekFrRXNXY3NEZmtub2hSY3Nyc3RWUVlOVlNWanUvc3d5bS81eXk4dEwwUnppUTdyU0kvU01jZHdlZTZGNklxZkVVVkJhUmtXUkozMFZoNjIweDRQaVYya3MxNUQ0Uy9uNnV5VUtreVppWk5CeHhyU1phRFIzOGFBRndJRlFpTU0zT3o3ME0zRHJDWUZyd2JpNnliN3RTZTN2VjRMSlRlMWtUWWRZTHgxbytaakcxR09BeVdOQmcxZVNwOTVzanRseGMzRDRZTzJhdnN3cnZUU0FZZWZZbmhkbTkxUXBqVjJvR0Jwbmk2SzhuRWFHMU12Q1laYXRsaWg2NzlETGJjRWdSeWtlUmpCQVhpejc3eTJyZ3BoUVVkamVzd041K1M3aUdGSGZiTUpENTFMazhFcm9qYmlGb0tGTlUzU1hhamVXTlBOYnViWXNUV3gzZFZLdWlmZ1FlMjlScEtuMWhtNzdWWlM5WDRhNzJmWEI0aGROTXBtZ042c0M4cjdhc0tDV1NMUkU4cmpsZncrcEgzaGl6NDNoemRlTU9IZVBrRGRMV2owV2prSXNyQTF0cnpnWkZTZXRuS3JLUkwrb0ZQWGJOakNLMjZjb1prL2FXTWZ6dXlQdGxZY0c3QndBckZvTzFwd04xOGZIaXdqSnVCRVc4WldMNytsZ3BIckFUS0lMWFU3bGFVc2kvblYrZU80NXAzWTZZOEdvc3pSZGVTMXlHWWZMck9JY3ROTWtkQk5TT2c1WHgxNllkOHVheFFMWjMvN3AvR2FsYzdZNDlHdmFzZit6RlZhWDMydDJ1aUs2b1V6RGtrMlgxZWN4ZXdiWVBQMlAyTU9GYUFsZFArUzN2bmVqQnJLY3IzcTI1N1U0dDc0RTBsWi93V2RQcnk1TDdWNHJPMzNsSmNnaGdiaGw2RWVIdXZ1NWNMbWY4d0szM2tEMUhaNVhQeG5nZWxIRDhHWDBDTnJyeERhaE1ybVZxZWZ4Vmdad2ZacUpDTGc3SUE2T1BoY2tSWC82UEJBdkFNRmYzamkvRnE3UWhKNmZLTGk2dzQ3Nmg1dlVGZFpMNFFDMmcwT0lnZlZrbXdMN1RMdWF0czQzQnFQWjNJSHA0YVBRZjUrN0lDekRXYWpMeHdkRUVJSnQ4UWthU3NGVW9SbUVpTkpzTC9wUHFGMnlwbVcyMnJVblFkcjJ0UlZXSUZscWFwUzh2SkplWUpHbzRNbTJXb2JZQWpOUktmUVFBSzNpYVhkNTVUR2JqVTVvZFhWWnpQMENwNGZFdnozcHdMbVNGQWxBWjlrZ1ZRWEZ6UEw3dXgwdDRoTVkraUROS1kzL3NSNUtxZzQ4UTdZRkFCbDVmaExiMTBqRVJoZ2lwQ1RMM29JdEVJVjhhTFZ4akNGVTMraFI0WGlxb2VuNXNjZFVUWDBNSnFrbVVCYmNtVFZ3MVhWU3JQcC9nVjRYRGdWSk5DWGFZbVk5UjhoQk1rVFl5eFJ4MVQ0bDV3Y2VoZkc5ZEpqbWVCSmRKSnZ2TCtwc2ZDY0h2dW8zOXp6a0pFbk5tcmp5SVd3ZzRGUE9Xc253Y0EwaFBlaHd3VlJhdEFxYlg1UjhFMFc2dThFeElLWXhicXBIc2RIU0JPMmx6T01Ga0lDMmcweXluT2ZvenhpbUh1ZDhrbjJjVW9BZEtaSkU4Uzc5UGJ3ZjhDMVFBS2hJWkc4cEFnbkdRWW5paWVYcWNWN3N3bTRwMDUwVlRuUEoyU3N5MVBlTHIyeE10UmVlNUdBOHJQUmpseldKVDVvd2l0RG5NTkYwNkppNGMvZkJZVHc5cnVmNzJJL0RIZFBWcnBwVENuMlY4THdqLytDSWZ0d0FhTVEzaDVoaWN0YlZ3SG1ZSEJMSHByRjR2VjJUcXhPK2M3SHhHd09nQUcyQXBGV2xDdkFYVTlZSzJwZGVTUVFWaldVWmNuRWU0bWlTS1IrUkdQS0JMQ3N2YWxLTDhmU09UUVpITEJrLzliZVFaRWNNOFBYSWE0QmRuOGxWUWY2Z29tcFlrM09LSVRGVFVpWUtKK1VjUHhGcXo0M1h6elBhd3Zqb1EyMWYxZHRoYUdoM1A3UU9rZ0ZZUk1BYmdjSU1BWXZmZTg5MjR1UU5pcEl3SWVDMDJ4WENnTk5ud0FBb1NNYnhXQ2J6ZC9JR05nM3J4Q1R0QUFBeVNhMHJaaG5TTWgxbVIrZUlPaGU3L2FQaGVzY3RqejZHeXVaZDYzVXFjck84Mkx4WUNmQlkzdGQ4WG41bTlRL09pNVhPaFRDUkFNWTVCbGl6SlliSllTK0xVaGRZcW0rRm5kUnJpbGtwZ2VUVGVhOEpXMVltNmFIcVlVV0RBbWlvRHFKZDloVyt2N3laVEZabU9vK3A3WTBQM1hhaW9iaUZKaW0wbkVPckdNQVgrSUFFVHBFL3BoMWdNWis3bm1tWUk3N2RtYkhONDYvRnVtNnBoTHlMYXFuL3JIenVBRFk3RU1CckVCUk1CcU8za3ZWaTdTTmZaazluWVF1WWdqYnM4cVl2NkxPeEJGVXZRd2JKYTVITmM0UlpCVjE5Q2tENXpzcllxRTJkbmNsa2ZIdXhUZThMR1BoT1ZUNFphUUJpMEg5ZTZ2TnpwQ1oya0pRMFQrNlJUM0YyU3pkL091WEpzTmVYazREcTJ3RURwTFlXdUNoUzYyNVBSZTUxS0R5YWxwR2ZvenNXMHFuWXJwWFMxeGVpeGZJV3ptem9hR0Z6dkNRNHZXYW93a0QxemhTRXRVVGpkWHI4MHo5MExidXdQN3dpMno1YWdJWXg4R0FTOUh3czJFSEp0bHFjNzFhRjdmcW5WZ3JHMlIvY3hRODZHVkhoeTdZZFpRQVRFS2dYU0FyR0RydG1JTVMvSHRPbUt0RFZmbk9lb05HNDZRY2JyQllsVXZIQWdoeUVRTnBmYStmaU5rQUFMT04yMlhCVG5BeTN1ck9WTVFnWFVrdDVIUnNrM2EzYmJKUW5tRTBHaW02MEJ5TDhhNFZ4cVczbXVQMFRqZGZYREVseEhTVFRsdlROTncrSlZWMTFqc3cwK09wd2NaNDFLTDBreWlVczg1eVkwUFFxN2V1VUtQVkI1Z3F6b2lRRHppM09mRTNZUmVTNXhZUWc3Y2R5a1JjKzVVUHk1TEF3cXFnMkthdGhEQnRkOUduQkVwbUJQSyt6bDNRMUpsV0xxREpqbk9xZ01LYmVSMjdWdkFId1BwU0lYbG9aVmd1RTNmSHNrc0lCMGFZQ0hhYWovUHBPNDRjeW1uZUhWYm1LeFIwQ2o0a2sydmZCdU5TVCsydmlDLzdneHZweDN4UjVrR0xtYVJlNnpzUW5GNU9CSk9hMWFqWjZiWE5XcXMzZERMR2ZNQWxoRFZIMFRmSXRHUjRrdmlJRUs1c0lUMmxuNyt3czNCeWJ2bkRsZkFyMHoyZ083Q2x6dkJUVGdtbWFIK3c1Q1NEdWZSRzNEWjc2a1FGUjJaSDh4VzVJL0hzOXk4NjlKaFJyaGRPNTVubFRvbkFxc0tkeDUySzluQVNVQ0ExeXlaVlduVFlSQTZtTExwTkxGazllVVN3US9XVDFRRExRUWw1VzdRMUF0S0VxVzBkTWdFWHVLbEZTUVJ1QW9TeXVxNnR6WUpRN0o1cGRKaDZXdEJ3REFuZTNpQWJtaE40SGJsNFpSSnFpK1R1NmZvckZaN25jRVR5M2RyU2pLRitjSldpaWtwNXNGdkhqeFRsQzNuV3BaMEJRKzR2bzd6ZHpHSzVsbnZibjJuc3dxYUgxSUtpQkZuRlJnTWxtQUJBNlZyM3lJaWxuV3ZtRHg3RUw0NDE4dVBZeThrQ0hBaEJ1bkZsTU56bk5DU1JQeVhYZThVdGRqbnBNTy9TTjc3VlhYM0g1ZUZnbHRXZHhYRmZvRGhjZnhHQ0g1ZWJwRE9tU0d6N01ha2ZHaTZLM2l1K0Q4c0VNOFpBR2tnN3V6VUZsS1c3R3JGSjFDcVQrZ0c1RWJnQnNnQnJocldFN2dtbll6c2hRSzBHYy9GNGxBYXFTSFBEcnMvaDFxcW9DUExuWjlQdkJDMzdyUWVBQTNFTkJzajBaZXN3emUxTEZZYUhDVVpXSThkTjk1cVdHQzVOWTlCWVNTaytJd1VVL0xCYmIrTmVJL0ZjakI4Z2xBM1dXUmN2MnI2dm93cVBkMnNaejAwZFQyTzFxMGdvbm11cjI4MWhBeFNsTTA0Y1huTVhpSTRYa2VnUURrYmRnV3Irc2hZMkJhRjZNMStGTHA0UTdLV3BvVzJHMzJHSkNtb1dWOUdrUVdtbjFoYzJrdUtKcS9VVStvNXF0U2xMTUhWQWU3QU8yS0U2d0YybUZMWDlFVTVaQlBsNmlITW1DU0JZQU1EMDFIUlVhVkR3Q3hGVkZSQ0NDdy8zM0hrOHlDejNTQmdFazA0R3FUZTRXaTVyTGFaYVlPQWdYc3dEZ2JabVFJMVh4cEQycENtOURSblJZWWtCaloyalorNjV3OVpyMW9EYUIvU3d3SE9MWlZKK0IxQllFcHJybkJMTXY2ODZFbEFsbDZZaktabHpwZW4wMjJTcTlMM2p3eDI2SHVsUWZGZEVnRUN6WTFaMlovL2FEYTdOZXVCSVJCRXI5Tk94SlVMTHd2T0szLzUyN1VIM1E4TWxlUU9tMmRhV2JFZjR3ZTdRNno5alpYWHVxS3pNbXp4MlE4d1Nhd1lpOUwzNjRtdStPMk1mWUZXTTNKVm9TWUtBVm1ta3ZwUVdzZUJnV2Uwc1ZudVdEQXpTTEI2dnQrS0s4S09wR3ltQkhpNUoxT2dvZ0VZS3Z1WlZYellnbTEyK0lIM0hpaHJBdzB6OS94VExGNFl2akhmTnljSzBVWWQxMFRKRVBSMEZYWXFjNmlOdkYvT2tZTlFHNXA1V001dFpycFY2U00xLzlzUGxHZmVGNmFZMThLVzNCY2h0ZzJaVjJnZXhpZEZTQkdtd1I1QlNsRkdEcHBjQi93S09STE9UWmtrUXhYMTZMNVZoY1BvWi90c0lwb2FCaFp6YmJHM09mcFdXc092YnhUcFBDY3ZFSjFlRlNXdTRQbEwzSE5pRmN5ayttWGQ3RnFqUXJiNkV1akJYa2hRMlc5djFOUDVKd1Z3bkluamwzVmNkVHhQK1g3VjhmYkNEV1UrQUZtdEFVOFUxeE02QlFxZHhKdkh6UHA4dkhoWEVwTVZwSVZDYTVUT090SSszZFJEVkRYNVFoZ2JpYU5oZ0wrMVVOMml4T1gvay9QME1JdzFMWi9TdzRjSkJlbTJBUmlMcERDZkNLcVQzMTVKMmJJM0dZbXp0YWRwNHpIV3FpTmRZQ20xemhXTG9IdHVKMW5vVzNrSncvbExWZnJ6T0l6QlBWWTVzODBENHRBc0lDYmxkNmU0S1FJSVUwVkN6TmNuVklkK0Z2TXA4Q3NKWTdFVnJ0SkFyWEw3OTRheDN2RnByUVRvSm5YSVdYQmtMU0wyS0tpYjB6RFNSdmNINTRPOVF4WFlOZlk2THlMN3lRUXVid0VNaThuTjdhWDhuTFB0WStxbU5FWUJSOTVYaC9vUEQzMmFuOVhqS1dJaU9IQjZSbDBrZk9EV1ZRVEdackVvK1FtVTBDKzRDclVpdkZBeU13U3hjZS9JNU5WaUpCNFRBandyclNJU1lqbkhjRUh4UlZtNFFzazFzQWRETitnT0txeDZSREhLK3pJdGVGSlUxWHdjcFlyeFZ0Nk9UQ2VNcXhxOFhHNnNBdUI1SmZCdm9pV3F4WFJXanJKYWFialc1MkhMWElMeUNkeGlBY1RpUytxNWlEVXR3UXFmTUQzVlJlQ1c3T3lIMXJ5Ulg4eE5XVFFHZHZNSVJWRWtvT0lpVWFqOWUyZnQ4a2w3dVd0OGVsUEx2cFlGcFMrRzVaUjhEd3B3Z3FFMnJOUGNlUkRuMWxGaVNRQTViSTlYb012a0RudURUeUJJSDFNbW9jSmhSWXM3ZWJ4SUU5d0xxaGxxRlJHS1VqakQrWkFndUhhRWdSNWlUTUlHRUprMDhDcjU4eXY3dDU2aElWL09tZTd3Mnc0V0FPV2lTTHcrRjBmc2grREtVUVhyQS9DTlRsSGdwVVIvVzBZTERpbGhBMGdLVEhIbDlab2tUQ1hsRTlXVVNkUDBhZlkyb2VQT3QyUjZvRzBKMnVBVUkrb0YzblNONEd5Y2taWmZLU0lQVVJaS29DRTBEb3BvMk9zRkxYS0dNQVhWdEV4c2xzdHI4S0J6RDhQaFdOSXZYQjlvbFpOY3Q5TUxkYjJCWjNydTZrZ0JEZ25YRXVDdUt1bmU1MTUwalg0eGJWdHZxRUZyV0RXeGlIb1NhRnZlZ0p0TlhZWWxLbVRqMlF3djFqMGJpT2Z2c3RFd3RrNnp4Z0p3V0xXTUFwemFGU3JHbVJESWRaVG03b0xtNHZ4QXRXbjVEcURRajNyMk9MQUNkazRoSldiTk9jTWFIOFF3T0VvVmZxT01OOU84dGJMZkVZUGk4WnRqVmZ5R1RFZjB5WFg4S0lZTlZYRndDMHF1U29qM1pHT28vbnh6eXVqTXY1NVJ3MGtCL3VrT0xxNEdRNlZCdFpyYjBKenA4M3RJTmw0bEsvRXY1bDNiWnhZYmgwbXR1TkhSVitqYlpQWVlMbWxyYnE2dUJFaEJsZGlIZ0pYL3JkS1RJdGNGeHdhdXhpVlpSWmduM09sR01nUzhzbjB6MCtVRFFvbGtJSkExSnVBSU5Td3hZRSt3S0lWTlZTeFZ6WkxnUktENUxUZEl5QzFvaVJ0UFpmRkpISS9WUjQxVjNjWnIxM3JjUUc5R3lLSWxSR3FodjVIWjJzSDRQQkdDdDVuWFdSWlNvQTF2UkZGS0g2bk11eU5NNmVzdkJQSHZ5eUpYUUpGSzM3VUo3QjNwb3FiVUxnbFc1T0w5VThSVzNoZWppYVNaTllMWnl1WE5zUXZkd3BPV3pNQ1dERG5KVi8zWm1CMHUwUjgyWHZ2anI0dTN3L092OWZ5VnBDMHc1TWg4QVdhRHRqQ2xjNWJ5aDdVeWc2UWpVTkdnWVJpbERRUG5pRXhsbExZYjd4RFB4UnV3UG5ka0RXQXFvR2pJR2dTU1lhR1F1ZzJDMCtGN2wwdHQzVVNUVjBTS1ZXWjJlOUJBc3UxRFU5WFFpbXRxOTZPUjhxdUFjb1dyNm9mTDZ0ODFFczExMC9WU0l4UTF6NUhJSVdxS0tOb29oYi9RZmw3VVhBNzhyOVg4Tk1tL1NWcEgyc1NGRzZkM015NlBTTlh4Nnp6Ly9iNjN0N1NSUVBLU2d6Y0pQNWhiU3oxYWNPbWU0R0xDTm94b3phQU9SaU5DZE1oM0todnlxMUVoUzhOZnVsbGRhaHk4TGFGUXNKdWlVdno5Tnc3aE4rbnhhK1RQKzRxK3UwbGRUanllc2R1bjl0OWpIWjRJL2pDQURXaWFyKzdSRVRUVzFJQWM1SGppaC9JcXdSVC9POEN0UjRIL0ZSeXljaTk1RmN2Rll0U1l6ZHI3WHNpazdCR1BDTlBLQkIxUlNhVWxNdDhRMjRqTnVvUWc1WEZGOG1idFY5c0VYWEE3c1RhZ3RndVFNRERPVy9pZ0s3SkNZdkVUSlppUGxhaWNlRy9QRVlqYzh5UEtTZjNNeXV2RGdFbDE5NDlmOHpNbVZvM0FJM1hlbldxbDlyWlpydkRXSDcyZCtHcW9QdHBaVGp4LzNkMkR0amJUK21yUzdkZGw2MWJuZVQ4MWwrc0ZPcWNRSFMvdDVOTlorbDlCVFBKTmZQOTB5bytReXhxSFJrc2xHZWRHVUI5Q2dsanMrU1ZBQlFkdDJFYVh6RHlySnNzMnpaNXAyQnBEQ2xqdFRJRm1CVWxGYmdSSjVFaUUzWml4aXJzL0k1ckhCVTB1ZW9lS3FNVWk5VTErY3F4NmkrYjh1UVNicDdCRnVreHlkbisxWUZnbWluYm1XQlFyUWNCc0ZqUVZwRE5GR2Q2NlRhUjZJdEpIUnBiWDUwUjhpVDJtV2NMbmNmdW5YKzdQWTN6OEI2WHVsNUxJOVlVSWZmTXhaendHV1E4UUZXazVaQWFodXFwSzN5R1dPMmlhcUpBaWFvVEo5dXFGRXVpOXFaREdGampEdDllYmdxTEtPT0JwM2l3YVEzSHV4dDBCWmpsSm9rbGwybjdqNmJxWDcrd1N2clZWQW1hdjNuRyt4SGo1SjlyRzlrNjRaZHhZMkpJZTVSMk5YN3lBbzNoZ0ZWb1dLRWsrZlZ2RllWUTYwWVcwd3l5bUE1c3BXc1pCdDZ4ZUlVTnErUnI5RVVUVVJWanRVdEFaOWdNK1dkS040Y3F5VkZuRjdMVmhwN2hCSTNLaWxOT1VCYWpla2FBelQydzBxc0ZEY1dCVmNZNHh2K1lBbUN2STJGait1RGVsbTZ5MUtMZkVHWXhzaC96NGNKckJWWnBFQnM1VHgvVWM4S3A3U0RhbklLZ3lNaUhLdVhjQ0tHcnhVWWNPSFc0NGtRbDYzU0tqVE1ITUhreGRmMlFXZys3eDh6NkVUWXJrbEZub2h1aWFBQ2NWdlV2V2FLUGFnSWE1bmtMQXArMllGRVhuN2ZqK3NoMlUyN1NkaFRVTk9EbnJQbEJodHJHeHNRelJXWDVReVhhRWFjcTBHVi9VQy9FK0ZVVVhDd3lrU3lGaVRLVzFEQXpWZFRGK3ZwY0hBZVJTRUJORDRTS2RGUXdYTHZreUM5ajVHc0hVNWR0K0Y3RExUN2RUejhVT3pDQ3laSG5pRUR6TXIyOGw0TWRrM2pzMVJtaUZmVVM5dktGUEIrUENuelZGR2YxWXFMdkFqelIyZW0xUEwva3V6Q0xqVDdBWlkraHhGM01vajdtV1JpaVNhSFBES01RZ2hpUEYyallCM3ZrRU9nSmlLcTBxUDlXQkNYeFU5TzQ1K1QyMTJDQjBNYTEvSEhRYnRsTTErbTk5dWV1cllFMC9zK3EwUlhYQUpXUmRhQXJsMDR3UVc5bzFsWXBqNjcxRmgwV0FRamUwaEdGdDNPa0tEQkdwRHFIcEpZeFBPcG0vdVJPMSs3RGZzUWFUMll5MzcrdWEzTXE5aDBtVm1VM2tldXcrVWk4LzdlWndEOVlHTktHRDlkbklSN2ZmUDdZcHpsL09UZ0UrT1FHYzRmWDBSNk1qcXE3S1JYM3M3MlNhdnpPbTNwVTR1aXIvOUpKamptdjVpeXVBamU1QkpnSG1FQk9lMFNoREYxQ1RtbFVoekdxdHNhVXBQYjNkeXhncWY0VnZxVEpscHFKcmNYQU8rTGhsZ3NhTHFzV3czLzRYWEY3ZllDclY2VjVrT0dJa3NQcWZHV28rMjdkUGlMdzd4bDBKYVRDZjIwclRiNWxmSHZZMEYrWjdXUWtEZTladllJWUI4NG5XTVFCb3U1WjlwaUNtVlJvLzhGYk84UXk4MUxxNVp5eXZMTU5yWnhJc2hGb2FHU0UremxaVS9xb3ZsaVhrYkxCZ1JCbU1XRzJ4Z05NMUNuMGJmSGkvTWRFejU0MlhWRGNkeVJwNUh3UE5PSURSY014VnI3Mkx0dmdLTkJtY0I2RUtxekpGbEVhdFhzeG0zT1dwZEtaalFrdjFhaWhJWHIrME9ha2NDUXRGZVlyd3dTK09nY3BSWHlCWXN0dGlpaExTZkVWY1BGZHI1UlZUc20wZEM5RlFjRkpCYUxXeWtSOEJlUWdSN1BGK2RCU29vNWM0cFY5S2g5NHZnaERPczg5aWtDRnVaM2k2TlltUjdDTHpuNGoxSDBuMERuNFd1S1RKQUpUQUQwTkpTWjFQYWhoVU90eGdLU2luRXJJZkIwMzJudHZ2VUtQcm81TiswUGExOTRUTUh1ZGFBdnJJL1hTK3dGb0Zlank1T0owQVpWZTFyZ1hRRnBGYVRjY3RIR05NcUtJbUFnM2R0dm1sVmhickRaTHIyVlFXOCtFbmQvNGRHRUJBS2ErSnZoelFWWFRhTVJZRi96NW1ta1k5eDljZy92ejRhSktsUlEyRVhsb3RDb3JwZTRUbTJyVE94aXo1RndRUkIvcWlJUkdZNGRYYXcxOUtDdXFzZXhDSHNLYURTVXRGUkk2TUpWS3RlY1htZVZYZitBYzA3M0NkQ1A1MXZpTmRybXlqZU9BRTVTbnFDUFltL1lBcXhLLy9xb1RKRzVURVlEUURVL1ZxZ0tGS1VQRnpjVmJjWVhSOUhudmJvdGJ2OWtnckZmaTlQVFBNdGx5aktzMGVsTmRrKzdoUGpNWk9BRjJDS1FCV1ZSV1RBUTg4dEVzdDBKWGJIZFBSY0ZucVRsT3M3cjIvMitVMkNuVmt0V3pZQWxnSCt5K0JiRkQ2VU5MT0pkSmdnTHZWdFJGNzY0a2Rob2toYzZTREFkRHp2MUNOQjlvREFSblFEdE5yY3lITElvbnlRNTUrYWpEWUdqVHFZSHRMQnFvRVJnRmRycnkxMy9nOS9lWWZzTFpYWHBXdWtrWS91SmNrRkhvY0ZndDdka3dVYUJNM21iYUM3Y2o2ekhLekpsaWxtRHk0TW9MT1h5dEtXNlBLVEFKNUV2cE4vZVpDS3VrMXVMb2M1VTZ4Z0pSdVgvMktUQlRwUmkzNmw5VTdHR1dIY1J6Q2JXa3dGTG5zTnFBWmhhdW5vT0V5ck9PUTQ1MjBMaHEybnBJZ3BxNDFZS3VDamR3WFNzV1doQkhkeTk5NHM1MjNWRHZSWlhpN0IwQmZhdlpYd1NkN2l0ZnhqbDdGUkRyZ1hSaFVqOGV3cXc0VmF0VlAxVFFETEExRWZWZkl6dElUYjBMcDB1cGFIS3dYN05TSkJ0K1dDWkNKT3JvZXhYUmNtaGxrQmp0eDRydGFnNTB6cTVWSis0WEZyRWdrRWRIdTFneFdzbVJyWUhMeXdmWU0wVlA0Umo1R3VBVC9CcTAwVFNnL2VTYWN1KzBtdGRQREh6OEZGaDRLMWpDczhHM3c2Wk9EUk5DaVJHeS9abDNEQkpEeFJtam96ZWJIQ2ZTZll4YlptSlVvUHAyTnpWekV1M0g1VDlOclZtMnZGOHdjSk5LTWFaeXVoTDF2ZWIzSG96RGR5OEZNVTF6bERHcUk2VHJkUkdib3dydUxXd3JsZi9VVWpPd215RjJhR2VZc2FVT2FKbEZidWNTZGtia2NJcXlYaVlFNXQvRVhINmMzMW9vS1IxQ3FEaUtHYVdXYkF6VHZlN0NuV1duZDd5ZWxBcFdRU01nU2kyVkxmenc1NUw4ZUJ0dlBTQXhtR3RUcFBjd0RvSzJsK1BpRmJNdnpWSVN4UFU3b0tJaEN3Rng5amxpd0RlL3hxa29FSTFlUVNaQ0pvSnNkckJTbE5MUnowU3hlOUwwUWd2dXZpRm5kSkttaVVRdHhibDIvWFFyNWpTZy9uaFFvUGtjTHJSTFFWZ0JjK1VFZ0diZ2FwcFVxekRnWjhMY0R5UFcwQ215L3M0OWlBOFlmTWFtUjVvc0ROYXMxT0d4VWo3cW9SOGtYTnZMR1hwQmNleVB3OExZR0FTWmtSUHo3bDZmZSt6cVRHWUZoWm9GU2VZeEdpVHQ3NUVvS1ZabUZGZDY4c3NPRGJKMWRRWEY4UUE3QU54eVE4T29DT01YN0tGM3JMWkxzQktFUVZoTFc3QnFUK09KWFBSVXRlRWpYT2cxdG9NNUlQeVVQTWRIMHY1UjJwbHIwQlgxckRQQkdQelZMcFdnOFVKdlR6M0t3dVVLcGlweDVsZnMyV1FBSVh3eWY3MnhWdzY0L0M0MUVxV0RPSi9NS3F0UVlCWDVydjI2TlBMVzlLdXhtMVQyZXdhZnNOQUZpaHRLaDBEUXlENjFORnNrc1B2N3ZNN1F3TkEvZ04rTGlMeG5IblNGWGlVN2hiQWVIZ09xVHBBcFNUaHlLUmNEN1djT0s3RVREUE9xcFZvazJTNW8zRUNsSWFHYklhM1MyMVZVc1Q2bVdhREd3T21UWTY0RTlOU1l3dzlGc1VXQlRTTTBGN3JQKy9YOTNFQ1k2NHVVUjRkYTNYa2lNU1VNUEVMUE1RNEU0alJ3MlpNNmJiTHBEeFBVbnZraSt0bjE3YzhFMkw4OE9CQ0diWmhFYm9sOGR4NnVhNU1vRSticTltZll3V0tTYU9yZmFKS2hNeVM3UUowMkRycHM3aGtUR09HQktXZExwYmVhaElCUUt5dU9uS3BkeEZlYzlKWGNrQXphVklJaTkrMHZqL1p3bjdiWVVmOXFQWnNreFozQzBxby9LV2hrMTZHSXNxMlVXT0w3NGJ6UVFobzRNS0ltRVFiQS9xRnh0YkI5VExLNXZLYUFyb2piQWJHTU5lNWFXdzhsTkdUY0FoL2NtR2h0MlFxMzlnRmFMdFljNUxYUmk2dkNjOEQwNm1RQy9JTU5YWVJxMXVvaVE5QjRIQWtoMHRGQ2xYVzAvWGExWnJhTElJMk9IY3hQOUNqUWVVY2NvUW56MHhKYUxwLy9leGZiUmhHajUzanQrdElYbE5IVlB2WVprZjhRY1VzVnJ2UGZiSTF6YW1xemhsMGFQTGdFelhxZ1IxcTNlYlQvcUFiVU1RSnFlc1FsenhmU0VrNUYrTmlvM3NlTzFjSVdCQmVwem5VWWFvdHZMSlpRb0ZwQjVIK3RCd1FLY3FxamN6SVhqcXVNK2craG8rRkUxMXlUdjJYeXUyd3c1SVU0ZldETTRBT2F0a3Z0RXUzTURtL1ZjY0xyS0FSL1RWT0VTc2U3ZlpOUmtHU25CcVNCSTZ2Njh4TVBVNEl2UVNJMDNuWFJtRlpVcVVDalNiRzlDZ3JGY29GSDBUNFJPa1N0ditDODB4ZzEyYnRDd293NWk4UEpvUmszMnVvN3FQcXhUZVRvWjY2QkJMQXJJekM5cFdNdzN0UTBtdDJ6OGYzRUZycStrQkJmYmFHZVlhbVlYMjFyVXFUQUpwWWF6UUEzc2U4U1dxcnZSTzBSWXNhdlVqQ1prS212TFBDU0JucU1ibThCS2h0dGJTQ0E0eEpyVXVEOFNWVUc3Q1YvUzBzaFRDUU5iTWRxSGoxbVZxRXhQRWFOdXVnb3Z2dE9rcGVjWDdJTGplcTVndWszSWpkUlVNY2k5ZGJFNWFMaFVtQkROMHk5QXhJcm02dVIzZDBuQVJHZHRsU1ZUa1JnUWc0a2pDV3hvV0k1cFNGV00rNXhFbXZoZ2hYR01OUFcxN1V1QlR0akVSeXZZRTR0bGQzdVBraDZuYmRnQmhwdW1HaHFQc0VQY2NwMTU3K1dsVGZITkRqWTIzME5WYzUyQWJ2VzZXZHhscTE5c1BTRS9qaHl4dStXZk5qUnhzQkdZclAxblF3V0E0NUxZbzRaMVhUVGFUWm9aMUcxRTRKQ1FkWUNhWUVmNk8yanoyRk1iWDI1QjRpM1NyQjBTSXBlb0c3YUhGcXViSGpya0oyTmhIblowblhwOVJGdXBXU0FyQkw4RERWZm9kcUphaWV5Q2YwQ2VSa1lDSWptSE95c0trY0N0ZVZ6Y1lhbHRyWEkrUk9JSWZKMURhSmxTMk5rV2FBUUtQS2xwYnFTdk9nWXJNakNCZVlwZjZ0QXNSb2xsZkFHcTFkbTJpYXlDck1KMUc4Z2JkMUxMOGR4WUNvMm9qSEl4U2ozN1ZxT3pFT3RhWTQxZFZoaXBjRHh0bHJ0UWJKTWk5cXpjQ1dxNUNobnVkQmxsbVdwWjJpSEJyb2VJT3BtbVRGWUt4RHdtb01wdmFZdElJUGpIY0craHYzT3UyRmFYTWpZVDZnWUdtQzFGOVkwSU1IQWlYMWpucTJoWWRpZGVRSWQvTkRReG9mQkszN3pxUnJlN1dqMHZqcmxrd05ST1ZKYURPZVlwdmdLL1ZUZWluYUNkMTlueDZXMHZKUzY4d3h2Mlk0bGNRYzBsZkxlREMrOWdpRzJCMk1UN1FLT0RPRWFrVWs1N2ZIRnlQMElMR2h0Ykw4UkhFbUV0RWluT3A3M1NrZUpVOS9ERHltVGJ0VG90c1FaTklqcmhZd3NMdDM2KzZkQ1JPVDJoelh6Rmx4clpybEhBeEFHc3pPck1xWTRzU1g4RTdIOTVuWW8vV0g4VU5peDBEWnlQUXQxYXNTblN5ZTBrVFB6Y2RhcFcwTEtNbHRTMG1MRkJ4ZlZ5WHlLdHl3N2FyZDUwZG1lL21hRldzdHhMUktlOXNha0lIb0lPbjkrWXprSnprS25wb1ZLV3pMdXNPbUc4Sld5a25qWEl2MGt2MmdqbEd0MFB3UFZ0UUQwZk1BM01ta1RyVUdrcGVrc01jYTU2SEZMcXNtNVdiVVgxSTF4dklUK3hjSmRzZ3FCYmMvNFBUZDZ6SjNWbFVsTysyTkh1NmNFaEtiT2hNcEZmRWVYU2I3TUc4NUFQSjNkL3Z5ZUFza2Znd2ZuU1hud2JSa2ZMWElDL0k3eW1FeGxyLzF1Qm1nRW0yRDNNS3pHcW8vMmQ5OVNMOE9zNFNhOEZ6M0FnZytUYUpiaFFVdlFHNVlHZno2eTVEdFBFK0FSeDBEYWZDMkRWQXd4K3gvZXBMTUZ5dGRwS3AyRnRpeDFRZzlMWXdmcmFRY3k5ZGxhRTBWckJlRDJpYkk3QUlNazlDb3lzUHdQeGpnM2FRbW5Sd0tMcVZnVVZ0cXQ0V1hpeW1tSTQrRUdsdjExczUyelZZVzZ6RXFLN1BkYks4cjRvSmJ4Rm5TYkNSdFNvcG00aG11cnJyclpDMUt1dVNPcnRtRXJ4MjIzdndQQ0RicTUxZ2QyQ1A4V0NxSnhTaCtlVm42MlhPTktYMHJSSHdnV2hsS0h5bVZ6OGpNQnpGSnlZeTNrbzd1WGQvZ0ZBTFJ5citNRUhlSVNnR2ppTzUwakFWNW1YalRBcDdHU25XdEpGZCtaL25BUFJzbUlGY1U3cmduTk9mcUhUM21aV3hSVjRWM3NIc3h3VitXb1dxeGlpYklBMTFKWnRQbHM3WnhmZjFYeHVhZnlxR0ovOTZBT0Q1VUZGZy9tUitSNDdQMTlaSkFhaHJ5QkZmTm9LcE1sVUpzWTNVRW12MktnNEZndWluNzMrKzdZRk90aG9UK0JlbmxKa2F6bTY3N2dCNTJhK2RUU0pNengrVVZIWVR5TU56N2Jvanhjc0J0YVYvcFZRaTZGZkZ3bUx3MUo4VjR2VmM5QmVkaFBabDRndnRPSHJvYjdHSDQ2a254cW9aVlVmeXJkM2Z2aG43aklIb2pXKzRlN0crUzBueVBSc3lEWnIzaWpQOG0xOWh2ZzYwNnVmSUNmaDlucUR1NnRrWEwrL1dLbHZ5bkxzbTd1ZExackZqRXBQQWMyUlFZZWNlSE9FWWk1U1V5cFBySmpWeU5lQW1tL092YlFZclFrd044QmpLc3BEM3VTeTBValFLZUtxcWtRbkF0ZUhzMTNDTEVacHZPRFlPTjNST2FSVUh0bzJuSlo1MS9oL09sdUp2eDFCS3JIYzMvNXkyWm5kODhKWS9iUzIzdzZNcnlFS3F2ZDgxL21obUR0cHE0bGJEVG5DSkNmL0tzVTFJMlJFWGFJbVQreDhnRlk1SEFha01hSE42VG9uRVV0SW0zRzFYZEx1Z2FseUIxWG9PSVVCVFpHb01LMkVmRGNZUnpxLzZvM0w3NS9LUHBrcURrcVFWdStmR0pJQnVZWGlhZHhjQWlhaHNqMWNZQ1JjVzhMVUdlSE9kOXNMU3dpb3NoNlNHRnNJeW1UbFo2cFBsamQxc2pIcU5rV2ZsWk9pdGZzVVRNTFA4ckEwdHJhOXNqNDRNdllTbVhacXZTbnM0THJvc0ZMNzUxKy9hM05MZVlXeFZQdTdIdXlkWVNOZHpENytXNy9DN0lKSTJTamlwOEZWaW44ZDd0NWtySk5GWW1kTFQyN2N2Z2hjMlFzQmFkdDhQdCtUdkxSMFByMWU0MCtzTDFid214MGdBdGx5am9NcjBRWFg2QVFGd1R5TGFGZ3U0bitkZ25GUW11aHdwc2VUc0NnbEZBeE40VkJkOGNnVzFBVUhQRjB1YnBmcnBtL21XaklMU0tOOXNPQWhacVl1WnZBSlpzMFZZNythcGRGZVk3MTNhRDhDaVJ1MXJWeDY3Rm1aUXpja0F3MkI2TTlXbC9RZFJ2Z1ZaQ1hhZ3gwVS8xeERSTDU5dVZQai9mR0MzK0ZxSThmQnREaXFTZFg1SW5MMEd1M2ZFelZTQXdwZy9WOXRvUzZSNW9SWkZmQ2VoQjdjdkRoQ3YzNkV1QTVxVlY0eWMxenQ0cUhRK2J4WVk3QXpGOXFhc1laaTQ3RnJ3T2p0a2lHZWh4S0ovOHlOd2pCMDdaandXdG9XQzZvOGJ4MmFvZ0E3VDhhQXlya0I4M3JGUnRGbnNqbENXdzJzaS8wODNrb2Z6NXdWNVlyUDFVTGEzeUt6VDk2OXd0NUxhR3VRMWtCQUNMaWJPTHNHTFN5dUE0bkE0TmlxOEpzYkRINlhhbkhWZnRjNWNUb0VtUUlLczlLRWl2WGFlQmExLzNxTG9lcFlQTFNVaEwrTWJub0dtbmhUUnFtN3c3MG52SGJwWWtnQXhxdXU2dmFyWGRMOCs1amtqd09pS2t0dzRkMU1LMGdsSTlrNC9pVXhXK3d5WFBxekVacDRJaksvUXZjcitqMkVWRk1selU1Y0k3Y090UU14eTEwdklqRmhJcThyV2RJVkZPbURQcWZJSExCMzQ0ZmxNWXZDNCtwTWw0Q0IrS21ZeEc4UTV6VFplNkc1RXVDeGxIZlQ1dUUzRkh1ZDNOeFlKaHdiWjlKWW9MNCtvL1dhYlQxemdveEQ5bzVOODYxTGNXMUhmZURlYWNvTWplRzJNSkF0N0xCWnArdWJjUzdibEc2RmxST0xpZk9aOWZxWWpTc3lIWFE2bk5ZL2FXZFlOVTVvcm9yelVxSG9xTGl4dkUwNi9HL0hGbUtneW1VNlY1enJINkM0V0E4ZndDQVNoQU5SRlN2S2E4OVJ0VVEyNWxXMmV0b2c3Tm5SaGV1OGw0Y3pxc0pyeFdXOTVKVXhadWNyTTU4ZnBxU1owSjYxMkZvR0grcUhRMXVGMEJnY1BrTHp1SHlKbU5ZSkhwVEoySG1CN0NQVFJQMXdJSStVMVk1MGNNSzVOeTcwTk5Sb0IrMUExeXgzdXZBSENIYnR2L0dQMkFTK2ZodWxHTC80enVIZ3FIRlhsVlNybUFLNkVtd1ZOZUlQOEFyUm4rTUpiaHRKZVBhZmFoOTROeHcrK05wL1lNbVc0L0F6M3ZQWUQweEJGbnZoamRSbWZqNnhvZTBRUWYvRUdUVW1kaVZHL3YyQnR5TE1zQW5qQ0ZWRHk0MWx4MmVQc09OVGdqVDRqWGwzUVBpWjNyb2Fra0h6N3hSbjRqNFpDemVyNi9ROHF3Y0Rzd2JlMFdQSnZjSi9McU9Kd1ZJbTZRRmVWZTdXeW4rMlh3KzI4SUwybERCL24zVUgydlJKWkErUzJ6RWV1YkF5T09VRjVNcjNkUFovQlVCS1ZibGcyRVhHV2w1ZW42V1RnRlBKM25jemR2RGVCcXIrWTRCZFZ3eEZhRldyeHZmZmxtN2F4c2xVM3hXNm42dExhTjdLNlFvOXJ2MVNwSnlld3EwZkZZUWpTMWV4aUYrQW50YmlBdFBkbFFNVVlsQnNWTUJtWm5OKzNiU1dVUWNEV1hpcS85aVgyUUdyVGZaNGNDTlRvR0NjMDYwTE9uRnNUNWt0anUzdDZ1b0hNbUhMRnRwRDE5R243TUxJVHhhVFF1TUxUWWt3UURENG5LS2FDZERobUt4bFdlaFgyQ0Jlb0VuLzhpY0pzSnVmMUhYVkxnRVpwS1NuZ3VTa1VWWnZ6MlIxalBHWkg2c1JLU3hBcTY0dTNUbXR6ckpiL2F6MGRsa202dW8rZmZZRHZtWGptdDNNWnl3c1p5ekUxaXo4UEQxd2Q1VXdEOXN6dlpsTHdlYWhmc0NBQTFsbkw5akRvbEVib2lJb09PVjFSWDdmWXpCL3h1THI1VllwcnhLTWtxdUYraHFkOWc3eTdHMCsrQlpmd0cweWZpdVdWZjU5R20wbk9rSmp1a0ZYM290Y0J1ei80THNsN3M5bG41M2V2eDVpQmxCcFp1ZW5RenlZVVAxc3I5L01BNFArK0xIcENCU20vQXVCc1A5dkl5TEl5Uk9tU3N2cUtncXhRTVlTRDVSTGd4WXZPd29KUzFwdWFKUlMzT1lDRzNWb3JZRkFnV2ZXbjlXaWEvOVNKcHJFVXYxMU1hMzQ2a2VCU1NkQmlOVmFJR2hmLytFdEJjaE9HeldHSjRYTkVpaXJpT0ltT2Zjc1FnL3RZelhibXlqdjYwRlhQSXducnZSNUYyVUw3K2VLa0x3dTI1V3UzMlJNcC9ha0NHMjVYb2RjcmxCU3Y3aU5JeTVhaXpBeDFNbWpONlRoUzFXZEhTWG42U0c5NGhNYm8rUk5OU1ZGS2MwSUVGZXZXUjF4aHg4ZzM2L1pvbFFBWS9vRy8yQ2thdmRmNFB0UFY0Y29NNHRlMFBrRTdRZ0JXWGZpdGFPWjJTbGxlRk5ISnVhVWxXbUFGeGFORG5nL0FwcmNBbkNuNGxCVVYxRWE0ZmdKMWhyMmxPQ21UL0dnU2l3VDRRb3kxWHd2RnFnVkFRVWVJdTBKNGJ1VG03czNWNWVHUndOVzJ0SmN0eW5LU3FKOUVhQ2hFVFZHQUZhU0ZRbUQ2blZqaVU4Q053RHlLV0lUdkRrOEtOTVhteDlkZ1drYXppOG9HSkZLTjBzWHdQcVd3d3E4YlJnTTBCWEptbGNuWk44SUkxdElNcEUxRDNRWEVkbS94Wm9QSWNXVDFKc2NUd0ZDbTNMemk0V3RIb2o2TEF6SzdtVkZybDZLNGRFSUNoNklIN3hRenE5ZXJWSWZMUWNsSW5rRlA1QkhUUEJqbzVBbERIT2gybGlIU3lqdUFmODArb01yVkJsdXV2am05d1REOUlaWnd6VmxjSXlxaitqSUNDR2hZakRwaEtOMFRMMk9vbWQveHNMWXcrMFJSS3NKcmxqQ1hmcUVFblYvbysvVWt5bHR4WGZMVFpiNHU5NTdmZVpyM2JRcXo3K3ZyczluSDNlZHdGUXpyaXJybUZ0bmh6NTk4OHJIN1ZWeWE5NlZ1M2VXZy9qMlZhQUdMN0czUkVmUDlWZWJ4dHhyK1VmeW0wbE1ySHdTKy9kN3Z5dTBNdDM3VkVsNFhyVysvNjN2dUtWd04vTnJjMkFMQ3I5cmVyWHBDM29OblJQWWU3TGJwTXZ4bWR6by8wNFlxK1YyNExFN3pFcU1UMDEwQ3Fvemdydng1bzM0eU43NzMxS25sNDJGWjI4cm5WOUxRYjF3TTZsbzNnUzdyMStLcURPRDRibkIrRmIraTNNVVVEN2RKenpXTHYwbDd3dUoxdEY3QmV2N3B4Ky82bTE4b1QrK1JzMnVqMy9PbEtUOW1LMCt3Nks1R25pMTAxNFIvUCtmL0JPSThvM2J2SlNIRmNKQVBGT1RsZTMrc2VZQUIxZ29IWWdNYi9vMEdmNHV3WmpmNzFxYml2RElZRW9uTmFWcFNLSGJLckZ3MmxGR0sxdktoMzVETS9Xc0k1em9pbm5vcUNNVWJwYXRwSjlIbkRYWXcrNlVKKy90YXZBTk0yL2plQW5waWc1Mkg1S3NXSXU5OWt2TjV0b2tWWTRHVzV5UktUaEVyWWMvY1k5UkRkZDhYUkRFblZIQjVVUW9vR3FnSmxRNnJJdEZoQzd4eDlPZGFESE1vRU5ZNkhtVCtTdmFhclN6cGN4Mkdwd0xCZEdxZkJ5ckNrbVRlY0VXQ0NDa1dhS3VZZDZhTVR5RkhOU2JSYnpoNm1wM2xBTHNUaFluUTlxamQ3ZGtyTFkySzFEVzVEWlJxRVNjdkZCVXNmVFovVTRBSUNmVzc3QXkvWGw4VEdLS3pGWWlqWWRMVFo3dDc0c0wxOVJORDhUbXo1ZFQ0cjBQd20vMFozK1kxczdHK2w1Y2MwMnBvRTlwRzFOejJaM1d5RHJYUUhyODRyb0MyZVovM3FyODJaK3Y1VG1OZnd5YW84Zkg2NnpITUVVQ0lCVzdkdmZyYUVURzc3ZlE1S3VucG51aUt0Mmt3QVBlenBrOFhySzhqYS9YQ1gzMWpsc3cyYkZDZTFJM1kwbmlhbjd4clg5LzdHWUd0T2toRitJSGNwM1NlVk1vdzVHU3BrbCt6dWIvekV5NitUSDVaV3IvZHNYYlhGcTRmVm41NUFqaTRlcWVpSXVxSDVzSlQ2dGJBM0JjNmYvUi9lbTI1TTNVTzRaWm50N1dsY0g0bWNtcncyQU1FV1pFTSt1Ukw4L0EyN3QyWm5aOXdEZG16WnRpZ1pFeUpPczBQUFg2eVg1WTNacmV3N2hjUE1wWkpQK05PanFRSWFSbldjYkwyMnRKR3AvakJmSlJJaHE4QjVwcytUN3JlU3U5Tm1CUlZoamt1WS96NStQV1lqUmhVS0tKTXF3OVVOR1VYanZtTGVBdXlLeTJpQTVscnprN1B2VzJOOGhZTWNDYUFKenFrd2xyQzhVdjlla0p1Ly93MWFPYXVxOXBETTJpRGlkM0lUZE5wcFByMDVxcFMzUFg1enkwdmllaDlrcDRkZUtyNnlUMzczbSs1VjlYM09pYitEb0V1TnNoU0hVTHNONmF4Rk5wV3haTkRXYWlVeEg4L2U1WEJYbWJ0c0F2UUtiSk9vRDlWOVlYd3NqajRHMGx4ZCtzbmhGM1NwZ2wwRTg5Q3RXdkFDSHNkM1dKNit3Zmx3Y2ZNOERyWjNhYXhXM0xZOTVhYnF3NXZwYmp3ZVc4R0duNUpmaW5QY2tuVUpHbVlEWGlNOWZ2UFEvTEtYV2g1cXZpNC9mcnFJbE5LVHQ0VXgvYzl4aDVYUHJlcmhNQmtxMEhjL1VMMmlxeTlXcHI0a3A1amZ0WVV4aFVIM0NRUTlFTEFwWm1ReGZ4dUxWdVd0MUYxNCtmaG5wNEZIQzVnTERRUVBwYkMyWVQrc1hkNVVnemtSZVZXYWRFcWhUeklHR1A2QU5naVRtVDZsa2hQNUxhWElxYUNXZG12MXlxQVo2SEszRUluRk9JRE9uWEg4WnJiL2x1Umd3QWFJNGprRmJUNTh3R2ZidDFPNjRlYmcyc3JicmNKdjlFZjBvR0U0bDBFYWJtMmRUTWRZM1BKbjZMQXNqQldOUU5qVjYvWVZUdlVFSG9jTWNzSWdCN0l5RlJrK0VzeHNhMndyT293Sk9pejhZcEVjc0xVNmYrcmo2cjF0UFdrU0tISjZtR0FRcFM0UG1VR04xVnlMK0tCWnZUQnZRRmtwUW9UNkVKZFFlZUsxWGZ0ZVo1UGJFcURSc0pTZWhFMUV5UWdXNnFZUkZtTWdqaXdYN05VcFkwN2JjWTNtNWw4bmlJL3VrZHRGSFJVNHdCVGk3VXhoODd3OUlveHdOeGxvVWJvTEh3d0ZIVk9CQUpQaStTRFVjaTVPMDZYT054YlBxSzhnckhNenVCcGl5dUs3c1BKdUFpa3JTOWRZSmMrblZscHpnVHZOR0FIWndiY3pmbDUwSWY0czNsTzVYUS9rTjNhbVlkTFhreXlWazdLYktmazJ6bmtIOXdERGVXdk9qbWZ2R0JiSFJMay9JN0RiOWc1WHpMeWpQRkp6dlplMU0vQUp0Zm4vUXkrNFllNmZMalROR1NhSHNmUExic1BEMHFzVTloZVAvc1A5U0FoM3RMc3plK2x3QmEzRHl3RjR4MkpCUEgrZG5NZWtQM2FQSHozSG5TM0UrRVRyZFNWdUF1UjJCbDRVaG1KWDlrVjBqYWJsdXoyUm5sSlIrN2VSRzE5TVlUM2h1NWpTb2xzOEFpODE1c2NkVmNJWDVkeVZmQ1lWN0xiL1c1T3ZCdU1JUWtYYWt0cGdmdG9QUGZyZmZQZkl4WHhZdDVnY2h1T3FpNnJlTkhPODg3dDI0Uk01dC85V1VWMDg1M0pmbU9ZUjZWMHRkODRFQ1U2ZnBuZVczOUlselZMaGdiYUIrdjhrVmRhMmN5UXhkQ1lKSXo5ZVBwU2NHY21kWU1vZGYrc0hmYTJ1ZXkvdzBYYWU3ZDh1Ry91VEwyaTNCNisrMkdvbFFxQzJQZCt3ZzMwR1hqVXRkYUNKWWR1ZU9CY0kzaXBsUGNjSFFNc0kwYkxxQ1hHM244bDV2L01hVDVOcjNNb2RZOWpiQXUrYlhPZEN1c3prSE5BTHplM3YvajlwMm5UTXJLRkZYOWplMm1pdi9TcDNEZWpmd21paS9iczB6WitOQjNvNVVFc3A5WGlaSThrR2orMlpheWFtc0Ftd2Q0VE03aVpNSTgvczh3dkRjOW9BWWV5ZWErR0hRb0h1T0cxcG5HbVVNaE1nWGlVRXJTY2R0UWFFQ2E3K0FSMVdPSE5zM0FOZzUrcndxbk84ZWZnaDRYd1BtamdJWlVGSiswREhrVTloS3B4OHBNWmgrcnk0Tm9ESTFwVnBkTXdzNzlma3JoYUlXTTZZVnZmMlNrSk0rYlkvMjVjbEZmNFRVK1F3OEgwMzcyNUVkSXNxazdmOGhENVVBTENPSnA0WjdrNjhxZ0ZvMkN3eFJ2bStiVVZNeTZmY0hGTVlxeDFjUnM2Sy9mdVhwTjJVcFliNHJLZjFwQlZsL0svUUloc0Y5N0p4UFZST0o2SWZMcU9VbmV3NWIzZXhsUUJ1RnoyaW1BVXJSRFVJc1BhYVBCR3lML3F1a3VoemoxeG1zcVlqaC8yVlo0ZzN2cmFCODU2YmJ2VkVEOHh5V3grNDRzTmcwUCsyeTZZbENBNW43STVCTXF2V3A2V0VUSk5lbWV6ZEhwQ21EYk1Nc3M5eG51Q1FmdWxCc2hwUHdPQVdhak4rQ1dLZlJWbzhVRTBFNU5mUlowK0dBZkIxMnFXZFc4Wk4yRGN4YndwQWY2VXZYN3dHczl1dkpuVmJ6d1hHUFoxYXg2WDNlcnJYWk12UHoxbjdDUjY3UXlYMzZBa05BTzY5TGxDQzZ4N1AzMjljdkNJZkt1a2gzMW1iSDN1V2taek9SakszOHRJMXQ5WU4rRERWUDdYbkRQZWFjRm1jbThxblBtcVc0akdwTUZEL3Vzbzk4ZzU3T3NDbUJHM0JzdWpCSDA2ZldIU3Z2blpKOVR6eU5UY1MvZkhrMnZacjRmZi9CWS9TeEtNdnYvenpaTnVjdEE3aDJmZ0g5ZGlWaHZSMXVwK0FQVkFSdTJHMnlNeWJVTnQzS05IRVlKTHVwUCsxUG1mMWE2RnNuaWpWcnBBZWRYc1Yzbk9LcUc2dWtBRWdKakFDWXAyd1Nkb1hnbysrR1JmcXNSKzJJWDVmTXdaMkhtVkVKcHB5MlRTQTd6dWxiVzYvdk9HMzBkczdmTGx6TXNuQXByaWhRNkhtSG1VTFlaWU0zeUluNkFoTElNQmdJNlB2SDQvdWhDNmRkVWsyU1ViSDhLSGwzNVJmMmxHWUo0eVRaSnV3K094MUZ3b2lNMFFzbENETVBaWmxPNHJMT1FsZ3dKUUFwcGwycWViL1lsOXUyWTVnb0FDWlNQeWpKN0R5RDQzeHlYK3g2QXYrNDhmdFFKVXVyeGVVNzVlcWczcnpBTUFEcHY2ck1MRFpYc1VWUjcyeHJCZURCdU8wa1NZRVpOcVhaMmEyWHQzMXpiNzF2N29YT1lrRFE3T1VJQloySFd0K2FHUUdMTHl6WlVyUWdnWGhYRFd6TkJSbloxUCs5THQ0MXl3emwwWFhiOUJ4ZFpWcVhMcCs2UWEzTXpEeDB4STB1SkVIL3ZleHE5MDFJdW9kYlFhelNSTTBjNFB2a21hNmFiSEt1ajFqYzEyZWpBbnE2RXNMb2ZEemRjOFdPUjdkdFJSaDEvRDhOTEM4YkxCSHR2QXIyVWI2S0JOOUt0RkhhbXNnM3JyOW1KUjFWYzlqWm45QmUzUTZjWXpDRHJoMU1iTThTU2lKSlZzVGhlYkxYcUhRZTdiS3JaREt2RG8rQkg1NkFqWGFNaEhZYU4zendOU0VOOFg1bUhXSWpXN25FNzJXZkZOdkd0QjlMSzl4aEVZODZUajcyS0h2YXlQcTN5NzU5TE56cTFQYVM3OFR1eGtaTVBDdUprc2p0alRkd3BmcHJ2Q0x1QnY3RlVORnhBY09TZ010ODVrbTM2NDhtZ0thbzBCVU0rSHpuZkVLTDVJSGI3endIY3RnODFZTWlCQ0lHNzBUb2ZKRlZ2clREUUFpUUdFZHNNRmpKL1JuT0FldGx2ai9Eb1hnOE8vSXJmWVFIZFFiUVVhN1VwVFhhQjYrL0ErWGh2V3Q1ZTZqZnZzVlZrbExVekQzTHNXSTBRcnVRNmJhdDVMWVJmTFppUnUzWldHS3IvMU9CV3Y5UDB4SnUzS3Rpd2tKZ0hsRkJ3TFFZMVJMRXRVVW9Ecy93WTFtYmlkZWJRQ0hyWXR3MGVYdUQzeHFTWSs2MDNzNHorOHY5NDZaNTUzejV1WmhwbG9aZkt3OUpqTFdPcExJMnR3cHpsbUlZUEUwcktuS1BxSXF1MFFiQTVxZkxEVGkvZ3AvSmJheDJ2Y3YrcDZkcHg1ZTR1alFCNDNnU2Z1ejVoc08zbGZMM0xJTVBCSURKT3dpOW5XTjVvL3oxWkNvVi9iay9wdTIrR0tORHlBb3NOcngzbTlOMC9XUFFqeEdWd3h4dFdkWndGWkhnQmhFZ0toN0VrNmE5RlNlYzBSdUtEVEVpNllHaVZORk9aZWNYdC94cTdqclM2clVneXVMSzRXVGdxOUJnRU1RZzNPeDNBeVFMajFsSFVNellMZ0ZKSjNRVlQ2QnBETC80QkJTeXYzeEVUejFrV25tOXpGQWhQaC9wYXNhQUppZ09KMWROcFhUV041OVlkWXIzQ0VITGQzMmVXQTgxM2lXQ0dja1lrMVBuTi9QK2VZYXRLLzBtZ1V3dUJvNHhqZDdyNEFDRDdMYW1ZYWpKazRzdXpjL3B3UW94WnB0RGRuei9HOVIyMldWdEFHaStjRmloeU8ySU9SWWZWc2p6SldCMTByT2tZWFpuaVhEQ0xobFcvOTJBYzBSWTRrb21OWXgyQlFES1lwWUtzQlNKY3hpZ00wK3RFR0NlaFhCbXFCMXRCVGhPZFR0UTE0MFhaYTlTd1IzblU4Q0tCdjVJRjhlWndVWUFqc2RnNVM1STR6R2VKUFEvNFZZTmkwYU9YTHk2R0JNWkt4U3AvTjI0T3pJdFBVc1ZJQXVWMXpmV0NFMGplS1VXYzVqaEF6VHRmVUEyN2dTQTFkN2x1NG4vNUhvYjh4amNyekhIVXJZVVF5d01mMEhwQ3NXWDl4am1acEZOVzZkYThlT2JsUHhYTUxUSElLVldhL3ZKakxKWTVmYW1lemJmeVl1Mm44cXhhQmZNMm4yanJWandzWVgvNi92SlhMUlpzcm5PWTNEMmZTb0VPSkpiZ1l6aXhwako4bGVORFJoMnFISmZHVjgwQlRYNkN3SDdabDc5L0pOVE5xWVliZUZSNy9XcWk4S0ZpUUxWZVBWeDBDRzM3STB2WGFxMDlpY0wyYWEyd0Q3MkxUZWxLT1BGK3o2ZlQ0NmNFS01xUmt5RVUxdEV2VWRPUUp4c2tCbk80bWZxV1lFcXdydWJzR0laSWczVUppdGV2dXlSa2NCUFlTdXBnNmVrbml0VUp1WGVhNktySm54d3pIS1JOL3hDbzBtSnRpWXBxUFNyTnlZYjV6a2J3WkpxcmkwZW5rVTJNb09Ic2RsaGphTzFRTUFmMlg2U0NPOCtMTUY3MzJzQmhvOXREVERCTlJvb3BkR0VzVDZ1TndrN3hqaTE2eEZkMFpXa2dxREgwNnVaWmxJd2E2NlpaNkp1NkpXaHVjUUFWbjVqN1cvUlJJcmdTSGdOTXhQUEFOYWpMSUh4NFFOUjNudWY4ZG81UUV1NmpkTTJUS2UvWFkxWnc2RzVXSkxwOXdlUHNVSit5b2l5NEJuVWlJanlIZnduNGVQbGtEMmdhTFl1a2tjbEdVZktRb0NrSFJwVzZFc3dDM1lXWGZSSWJkWW1PMTlSbmF1Z2l1VGFEbjdWWm8xdWx2Mzd3elJYNFZibnc0d0czYW5FNklYUnI5eGpkeVhyczhOVFhRWnJ6TTdJaFlNcnJTdDU2d3ljcmNFNE05TURjVEtHaFhBeTBXMk9pclYrTW1jOGh4UG5LMmU4a2p3c283cHlUSHprNHFQL2UrVUxvaHIyV2IwbnpMUXYvRFpqWWhyRTZMQTNIS0duR0F4WmJ6N0dyZTBnTXhZYVFRQ2dsNlJabUxzUks2ZVVpbndPa2tFSmE2MFNiSFNidzZMZHBDb1lVVENicmtzQi9yM0RrOW1xd05LdCs2SnpxbTEzdnpMdkp6aGlMY3M4Y2NJMDlURXFsVVpaVzNCOUJKbDlISWMza2hkdWUraTg2M2xMeGNJanhIOFF1Ymc3RmY0WjVuZnJncDNSZXhiVll0emQ2V245RjFLRzA4VHVPSlV0Qlg2NWZHQmtpV3ptQnB4akRmRkUxU04xMHB1a1ZBOEpoNDlaMmwrQW9YYkZLUjVaMk1JeXgvc1JvRHpDaUNwSGlkd205MUVLR016anZBMTczSFNTNUs0R2Y0dEZ0Rm8wWi9ReXhKQVAzWlVGbTFuc1JVTG1LSG1yODhFdDA0TzlUV1N1dmE1blYyczAxQTRBUk0vbG9pbWRhYmpaUWJiZDdXSG4yMzkrb0tlZVBTRmhiZG05Z2RybzduNlBYaGhmT0oycDJRYXJHQkVxc2xlMzFFeEhRY2JUcDcxa3M2TExoenVZVm5adVh2NFdsQUNtTVkwQ29qQWVkdmYyU1RHMVdWNkxaZU4vdkROcFRMZExqbVZETndxWXRVdUR6NG01YnZDcjhxWW9QMEZUbDUrMmRqRDhxNm9tS0E1VmpHelZyQURIWlBORzM2SENBTmpMUTBPY2E2elRVcFdXRjhBYzMyOE5MVDl4UThJSHJKQVB6TzFNTnpLbUVtVXhGdHJ2WERtTXQ2QThrMFdhZWgzSFhsVDFaSGVWeHZuUnA3KzdOSlA3NVMxRmFkWGFDYnVqUStjR0hEbTYvV3Rpayt1Q1p3N0J2ZURNWW9tdkNMenlydC8xSFpUSmVZbVBydVAxYnNmMEdFZk16ZWR2VmNML3JoeDRnQW52K2ZWbkorZUV3a0Q2OWFVblh3WmhvNFBFNUx0NjAxRkwrYmUrSDZoZXNIL1gzeE9wWjcyWXE2dFpRQ0lBNjJWWEdzVGJXVUJRRk5ycjMxKzJ0cTIvV0tVdEdidS96ZmxrVStOM1NnZ3lOQ0hTNWNlUlFRQlBpWnVoR3VsNFc1NXJCaFJ5dlkrVTk5UzBmYStkVm5acDZMQURXZVNGOVgwdDEvNStBSk1jbVE4Lzd5ejQxMWVZelVXYlFueFhDaTMxS3Q3TENyZTZTNUM1akVNTzhtWjZxZWNXbHE0ZG41VTdQWjE0NTE5eEwyMm43QTFWMFZIcGdGTXNhZ3pSeU56anFuTTQxM1RuM3FHN0hEQysrQkNPUXZjVlZ5U0JhQ3QzUHdzODBSeG1sMkNxS0MxWFVvSm8yd1c2MjI4ckJMWW9RMnpwNHNGOUVtZDVjUjF3WUhmQVVlUFlYR0tpaWZVYklqWEd2UXZwSUVmbkVaZG1lb25GcFpsbU9iNnk2aDRmczlLVHFUM2JGRUlRUENoa29CaXJVTXFnTVBSR2d0SnhOU3RHM2laQVBQU2VGaFE5MklibDRlUFo3bDZoZXQ0MDExelBYTVpvUlIyK2E3SVNGMWxKRVZnUzFSUHJDUEZtR0dSeEFuMm1GY2NWdmI2YU1ycGpjdk1mYjZrdjd0Ti90Zi80bWRHcjdhd2dlRGY1U2NMZHpBMkJkWFdMeFZwL3BxSC9iTy9WUTN4b294WThGRTVFdUtsL0c3aFNyVjFaa1F4Tk4wWS9NL0xzL2lCVko5K2MzYlY4THVZL25lU0lqZG5FeWVpa2FSeFgyQTluYWFyUEpxTFcrY1VMdDN3VUNhZFI4LzJJR3NtT3pMS0Q2eHowRVYwU1FOb1l2MS9KdHZIUjQ0V1YzTGlQUWx2azVZOUV6Ync4eGc1UktxckoxOVg0SGw2MlowMlpabXMxTnkyMUdhUmpqVlRrVUVOTnpQWVJGc0ZyeDN3STBIUDEzNmZ0OHgveVFSMTAyU0h2cGJudVJhTXdMbnl1NjRibFhtSE5OZ3lEaU42dlE3WmNLblIwNEhqQWN2ZG1oakFTNUFZNlYxODJWcjRsMWJZL2NJY2VrM3V1WGl3SEpuRmVsUTNSUjF4cm10SzZ3cFJWSUhXWXlFbFdydURpV21CV01VQkswVUtQVjlFQXR3NjhadnpVbk5CTEVJVytrQWZKZEVid2dJWDg4U3BsOGJkVzNkcmlBbFlmWnUxRkI1dExRaWtPUHRWR3ZLOUNDVDhzY2R5WkNNTkxUS2JuTHJEd2VpLzJWUGpQR2ZJZlhvWktPY1VzczBnVmpkTGZ6WnZsZWNPZldBTnN0Zy9VeVA3MVowL3ByNkl3TkRzcGk1NmxjWXVQK1Nldm1oeHc1aXZFUWNPendmMUVnMVBZYlY1S1RKM1prUklSL2wvc2ZrU3BVNFhEZGFLcnFWUmhqT2RvSkpTN2RXb3FKUm5QK24xWThvMWhJL1QxeDdic3RzeklDQXNxakloWXZkd3hkNFY1NXN6RmwzdmtTZmpscVQrbGRPQ2ttY1E3VnBIUndKZ2x5RWVBWGZDWUxjaFQwZW1pbGhXVnA1ajFHUnZEM0VRS2o4dTBWU0ZQUGd3NzM4Q1RqN1M3V253TlU0RzZDTi9tWWF0eVJLY0lYZTFRUFM4U2x1d2FZZW5ucm42ZnNhY0lNZVYwaUxQd21SSHU0SlFGbVludHlhVGVYUWNvVmtqWWZzZUo0Rm4yOUVkYmoyVkRDNFNCRUpJN0hHQUZkdGI2cTlNdkJua0IrS1lQODlxS1FPVUowMnlYcjJ6cWJkbmlOUk0vVUNTNkpEN2tSY1U5WStkWTFNamNybXk4SUE5aUFTZ3ZIYmRsM0ZRSXdqaWFqZ0hIZFZMdXByQjFZTXVsTEQrZ25FQm50MWNhYmg1Z0NwcUI0bjhJbWx4bnRobUhnSHNQNEJwOWpWNW9lb3Q1cldtYm1abGFaOVBoc2hWK20xSktlYWtyS21CMFBKR2t0SFRMUnFySm9KejZveENBRVJBSnhrdEgvbFZPMnpTWXFyWlk2ZFp4YnVnaVBoZWNUTkNmK2xTTmdrMzVoL3JKQ1RVQ0pYTGk0QWo0TlJMcExPbUJPQzhVVHgrdzVyN1BuY0NPR3VmS3oxMGFRL2xzNGxUTWJaWDlMZEY2eFk0Ynl1a01uRjZjS1FNZ1hDMTJnbThrdGk5YnNDZENWQkxwRGNNK1d2RGVVYXY0QkRCK0l6ZlJlZXo3NXZRWTdMYitUUkJOS3YzM2NVNGtBZWpqaDhValIxTGg0NHFLWHZNZTVUQWhTUzdlOTllUDR4MGllTk5vRHk1M20xMCtHZFVFZEJHWXNTSi9jYUVoRnovUXFka2lraHVTUDZVcTg5NjczcWR0VSs1TkZ4S0kzT0F5dGJoVmVjTFVoRExHSEFDaDNKTGpTeldhOEgyaWlRVGRNMDJSQjdVbE0veDhkdWc5cFhRekJxcyt6YUxCbHA4elF3NmtTbkZKcUVYVVRjVHN0bmpRWGRUc21mVEZ5Y3ljU1RsdEc4UmtSOE5reW9lQTFiUDUwR1BXWkltZFk1Q2NTZGk0Y2w2Sms2RnNpeGhaVXpHYUFRRUhWL0s5cDFncTlhWFZUM3RhS1pFOVZ3MWwzV2F4b0ExVUVheE5uRWZISW40SWRpeTJseVlXN0ZHQnNtWkdLTm5BR1lQa0U4VUpIQjBRTlhmcTNBTCtaekQxbUdvREFvZ2pBZXhCRUFNcmI4ZXgxWlVxM1Z0Rk96Ymxnc24zTUFocWFEdkRYY01jNHJsZ0dXb2xXN2d1RVpQWGhUeWZIclhsWnFHRVNweDJISDIxUVBmMTlnMzNGdnk3RTdMQXZ1TVUyTjJ6MkJFRUFOZks3TGwwR1MwelVHbU9XWlhoRy9zVWg0cldqWXozUDI5OXJrOGtGOFJSUWV6azhIeHNrOXZYQldPSkh2K2VNWjVVeU5EZlU0TWlCZ1FGM3NIUXVYSjdQQmpHKzNNcUNTV2dYTTgrdU5pT1FWdEV3WGNIUmQyMnp2TmFtS0RkdWhRSTVyZGJtdWRYOHdMbi9jQXZWMW03M0dHK3ZLT2F5VTFUY25jTUJzanhDanc3UXkxU1NXaGkrZ3pBRldyb3pBSjdIZXE3ZmEvT0RGczU4N3AwUUM2cEVNQzhPTTlsbVhVUndEdHYvSjhHVmswZEdvTzJLT3F4OFNYRi8weG82YjFhcnVjMXNYNUVmdytrandmWGpIbmhadm9vSWxLZjV4cHJQR0tXNTViT2lZYnNaelowckRLak1pYWloS05BVXg3MnM0dVlKc0h5ckdYNWZsK0l0NnpjOURqcG1xTEpKNS9LM2pTS2EyVTVka0ZIaXRYNUJnSWVNNFJJQjA4NHhqdFR3L3N3V0tNS0RFNWM1QzhYaEpHdlBQenh1Uk4xcnA4QnhESEkvU2FtTGpkRVJYWjVIRUNTU2FaZVIyUjdFZkZTOGJCejRpc1gxYVIwOXQwSGNpb0xSbkhnSGptQ3R2dy9QVDA0UkRUeFVtT1Q5bUE2S25XUWxXQUFwakVFeG5iSHhLdXZqeUltdG1IMW9XdFBRWVlkakQybjBWVEN4NXUxbzk1dnQvSUtOZE5GVTdjNzF3K3pkZTNWQUdTbkJmU0ZoV3ZobUJIdnRwdW5FQ3ZJQks1L3lWb2l6eW9nTnlNa3hYMlZuYTNVVHhPc2xnZE9na21aMlRzQllDbnBubWFxUlVnNGYrbGw3NDkrTkJBUFY0enBDb0c1UElKQzR6eUlCK04xM252dkZEUnBTOEFqWDkvM2t2KzQyTXYrSnRWbUJ2Z0xWY2RiZ0JRYXhxb3dzTndKZzdVR0tIS0xPQnRsc0hDOXlhTlYvTGVkRnZCNTU5a3pmdUhPUHB6NmlZSkRzQkp4bjFDWEpEc014TW5LZTBlblNueDAzdGhNajNSM3l1Unp2emF4RnpLcG9BVkNlYTRaUDZVUXRERkJTdFdoTTV6dFlZYVNnUXRPRzdjRkJ5ejFCOFBVODU3Tm1IeStlMVhUbW9FRkFlaXB5M0VacGlJalZRUmU2dlV3YTNBTHR6V3duQW4vZHZId21OalErOHQ3VnNXK0J6RjVqL0hoSWcwRlVwQ284MzJranhMRFBKVWhZL040dXRGRmF2QTh5bkdHNWV4aVBaNEV2U3JqWVF6bHdwRVR3SGJpY3hMdU42NkdwL3JHTDJOdEVxK2NURXA5dzR5K28vbmpHRyt3Y1RUaStqcHFzT05GaWFKNUpwNXgyUmtPVDdXaHpXN0Y4eTMrNlhmVVpWa2dWTis4ME1GSVVzemw5bDdCMDBCQVAzU3VhdVhTdThxci9RbnNNeUUwMkVhek1mMW5GTEdiczM1R204d2gzekhtN2RnaTQ1UXR4NERkY0hEbHM0UDZHV0E0R2ZPNitwMmRMSmVNaXRBWlp1cEQ3OXdvL1U4ZDBjeXRaYWtTYUFyd2kwWWZEczZ6K3ZFMGl6TWUxeU5PblVpNGUwS0xrd0RkM0M1VXlZdDJQSzlDdlhFdUpjZU1EVkVRRW9Vano4TU1RK0dwTTFtekJNQURxK3dqZ1k2ZDRJRFByeEhDRlVZL0p4c2d3SVpodnhGNFVBa1JrNGZPdnRFRXFIT1czTTBNRXcxbXVQclZDU3ZuVnROcDZDcEtyWWxuNHJwS2dGZ21uVmtxYnNKbFkwWTZWN3I2SThUSjNiWEN3d0pDK3lhcW0wNU5nNUMyemdkV0dTRzM5cENBalFWQ1hwVGsxVFV2bkU5VzhhUjdQZWVOMWNRSUxpL2hYTVJDeXpPVXlRQmRHbktoR09ZQ1pQVW1TdlpOK1l4REtCWlcyazh0R1h0aVh5N3JGNDNjZm1vUzM5RHRxOVBWQnZ2bEVtOU9IcUgyZjBxYU9GQlBGQ1JxeHczZmd1QUc4YkpMMVlNbUFZVVVhUGVMK2tvQi8rTVVaTGNMejVRSTRDZEpzbkttMkx0NlQxcDlORjMxekFBQnZ5RjBDaTNTVm5Dc3dRd0FKUFlBTGJsb1ZzcDUwaUpkZ3UyckpVVERoMXN5VEx4SnBPNmlwUHVKNlRBNnB1QzJLc2tkZnY2T1RkdkJ2SldFQnVnRXBSVmVFOTQ5bWdpWWRaVWhhS1ZPSjVVdjB4WVBKOTJuallsWEFKQSt3a1ovc1E4UHQ4NWFxV2xvVC8zK2p5cXpXTXdoVWVzWHovTERUV2FKekJZcXdHNnlNNk5nUklmVzhlWGJya0k2cm0zSmpjdWw1cDJUeWpnakxMOUpTK0l6SFpFWWpMYmxqLzIvdGZMT1NrYVczWUFJRTdXWlgyUGlyVDl4TFNLRVNYbW4zenRvL0V4TjJSOUhpU0thd2wyUkdUM0xYK3pwV3NveGNSb2JDLzY2YlBpV1BWYmJVTlJBTW5HdUpJNFJ6OTdUd1hLYWVxck8yTzUxdzFqbWhuTTlIOWhqZU9CcjI2dndEeDV5LzdRS0J1Y2QvQk5qak5qRWZZNE9sT1N0Kzl4MldFbUQ5b2RlRSs0L0xEVS81WjdyUkFFMnQzbnh1S1QwTmNRSE8vZXNja0VBQzZISWVxK0ZWMjBhcWUxRmhBc2lLSXRTL3JXU3hocVlmcWlWQ2F2cm1SWk9nNFJnTHkrQjBHUWx6dyttRWJiWjc1cDA1V3hNYzlmSUhyQUV5Z1dEa2luTFZKZCtQUG5FSiswd0crZ1dUQyszMjE4TTg2d2ZkVE9mL3lCYTNIYkxiTDlDbFR3MmJoTHhSVDlOcy9zZkFWVHlKY1RXNjBlOHlWU1NGUW9NQStGa2taazBiSmJSYVlkT0R3VGwzakNYVUZOOHlzbGdma3Fib2tDVENLUkN5cEY3YWNwMFREcXp3N3d6dUtEM0Vob1pSUVVXRTIwbitpWlVOTXVkUXo5dmRuR0NDNnhnMmthM2N4VUtXdFNYemx2dlcvZ0t5QlZXajFCSzYrODA5ZXd6cUJJSEZybmRNU1VvZDRwU0NaeTdRYjZmT0xDeVhTRFM1MUFDeEE4dlJyNjVock0wZ0R4cGdaYVNUWHU2cktOL0Fyd21RbWFqZExqWHFsSUZFNFVRN3E3MG9MTWhsUk16SVZpYjlYaXdQaW9hSWJ0V2hpKzBpRHdjZVlUT1R5VVNJRjhIcGRFd0JDUmR1bTNZZGFaalFqd2JJeHBjTDQ0LzdWWVExekJDUXlJZGVOVFErZm5GTXg2QTh1anZuREUxNG8vek9ac0tlbHFETDVVejZ0bTc5dTlSWFhGSnk4Qmd0M3FvdS9lMGlWUU1WUTZONjB4eE1ZL0l4OXM4UkM4RlpNMEljQnIxMmRXN3ErMndiMWZkT01yaVFGSVZHVVkyNHY0MGlKY0ZDOE9mNzA0Ym5TSXFESnpkMHkxK1R3NFpvcG9obW05cFluZ25XMTlCQVdBTXkycmJEU2lVSE85bTE4ZmtYUTZlMSt5YkI1OUdwQmtQL3I1ZjltdUMyRUFyUE5zdWdCcXBscHF6UFhtSm5ibXhRM2tYek01MWlhR1NxOUZvY1QxSG9RN1V4TG5QTXozcjMvcXZkTSt4M240ekVZZ3k2YnEzVWo3blZxc3BJZmIwbFdUUFJaZDNMbHF3eFVJNUZ2TWRyYjlJSktBZU9ZdU1lUjdLa0k5eUd0VnBGQ0FwNnM4WW9EbnhoL2V3OEJqZ0FncEVDUXBkRSt6WDlTK2Q5VlV3dVJhNVVrUDc3MCttWXRJSFU4Q1ExUzVtN01VNEVqTmtjSDJoYldlZVEyUHZaV0pWT3g2YkcxSWZELzlXWVhleFFHK3pyUS85Q1JreHVwSGFFN1ZQZ0RGYStZRHpQT1M3MXgzeTBVKzFZNXdmYXJURDN0alJTNzBCdGp0byt1R2FUSE1hWkpRSHc4bHV6UEd4c3VxWUtOdEtPcm9oRU9xSTh1OGxYS2d5TWlDV0x6Y1k0RU9FUUN3My9zYUFyQjUxQjFieDJvMHNkUGRmbHJLVjVoNlRWMWU1QnRZYzh0RGRWc05qUkRtSzFwSjZKMlM4K2RybXN6MG1LUFk1OGRWSG1HbDgrOEJVbTR6bUVtNDkvLy96RnJXM3RUaXZmL1A5RjdyYkZ0cnFyaEZxY25ZY0FGSEYvcUttenBxN1EvcElRclFWdDVzTU43Lzh0S2U2cTg3QUxDK2RJWEtZbjI5VnN0WWptTUVncHY1emRNZjUvZ2lSLytpQUlCUGRYVU9sUjZtUUpSeTZsYTZkSUxwMndCL1NpeldyRkM2TWhsbjFqNDFvaW5ZRjZXR3F4R0FZMG1CbFdQSkJZbFpybEt1YWJMNXNhYU03VkpIUitoNlZ4Uzk2aFI1NXQyVi93RTQ1c0gvb1pVUU5IZmdSV1Q5Q2svNnFHZGdHUVF1SGJCd0JXVytYTy9ydTNnVWdBVitvUlVRcjZQQXRaY1NxbkV6TWFHSTNOQmdtNjlFQ1RIVStNRmdYSFRzVXZQVkxIMHBDeWZSSGZGUUxKVEVwcnpHVFU2ZUR5M1lwdUVwTHJhOVdRYTRpbGwzeEROcDhxb2JIY3hjeS9DeFAzVEMyTE5MeGV6NG15Zm5EUlo0aS9Xa3p2c2Foc1czbVJUQzhvMGtkRWtsWDJPbUZ6S3NscU9jNzQ2MXVQWHFpYTlWaDBqTG1ES0luckVVUkdyZS9ZNThKdWxaMGppdHdzSEMwbW1xeWFPZnNkOE9Gb0FIaVNDL2FtSkk3VnpDMmFQRmJibW8yeGRpdnBVR0pQem9PejFlaVZDdXJ0OTMrMmNlQnJvMm43K0JnYzdpdWUxOHN5TDVrZnBKSjM5ZTJVNXF2T214bW43WEFlQ3VqczRubjZyT0hvcERsU1pZR0R2Y3VkSks0K0NYWkl4Uyt5UVF3T05qajU0MTcySm5aaFJkbHNHMXlsTUEyUHJjcE92eklnTGx6aWNESzlwNGF2TWlKMHNBTFNSTEZaOFdEdDdjSEFPTTVGYjY2VTIvTTBHQXQzRlBxMUJkZUdyN1NWQnN0V0d1V0dlMTljUURYK0s4eEY3SGwxZEFwMEJyR2RDUVpOWnlaTFR3UzdObDVlVnAxM2dzbWdFcE5lOTgvZS9xQVBacjlUcFhrbWJlTzkrTTRERGRqZVY5RXRIVGpvR2VGK0tyVHpONnFaZnVFdnBOQUt4SU5qdXhLYkwrSkRrU3R6MzYrblNRbzZjOHVURlljK1BJTW9ZcU1JOGlnVmI0RFk4WkhFdHpuV1hvN2hRVUVLeFVHN2xScllPQUpsRDIzWERBYTNDQUVpQkpPM0c4YWVBS2NpYnU4Q1VsWVE3KzBLOTIrbXIxTWo2OWRNNmkzMXF5N0RNRE1BM0xIRkhDcHZxbzQvOXNDTlJzd0dkUmFYb0M4TjM1TmErSGhRYk83cnpkRGNFVEUweVY1UkRFckp6TzBKTU43bmtyVHZpYlRLa3U3ai9WTTRhNHAxWitEcFNCMkFNYTc3TU1EVjhDb2h0aUVWSU1wbThOTVprc1R6U0ZRK3IyclFER084QjdZWDZiQThhODlGS2JmMjlnZXROc2RVd09CRUM4Y0cwc1I4bkVNQ3NPbXpobzhzMmtiOTM1MUF1ekcvWXhSTmZkWmROL2tTR05BNDVySUp2d1VZZkRhT2krWkl4dksyUnR1MzNLZS9nWUJyWktXZGN2bUVFcWV6SVpTSk9EeHdQMWdmdXYzMFMyMzFyM3lXamVkRG41dHY4LzVCcFAzQkNZOG4rWHN6OFdjZUpCREdRQVVCQXB6V1pXU0xKcWpqcjYwcFRSUGYwZ25nWGVMbjdZZUtqSU5YOG9KWmpxaXZSa1BLR1BYYnkxdlRBNllVcEl2QXdBdkxPZGVPVlpESWFacDFLblBMYjdZSldDdmh6clZPYzJDVndMUHY0a2xMdVQ2YzhOV3VrcDVicEZuMW1hY0huNzhBaWpYSW9iL256Qi9vMGZtN2RsWGdUN1pNdWI1SGYzYXVtR1hEVkRmMzZ5TU5ZMlNSVGFYVU1tVEduTk9LSlpHNWgzd0Myd05MVElWbTkzOStGTWhEbGQrbGdOMTJodlQ4ZkE5Tlp4TE1PcVhhd2UxeEtKUHppZXlkcTRYK3FyWmpqWFdlQm9YRkNqemJESnlDYTc5ZHdBWnZBV2s1eFk4TWlFTVlJSUkwaENvS25wZUJCbGg5WU1za1ZYdXJaWGxNMlo0V0RqOFJvbDV4QWtBeHRTS1NmZ05BUUNBWWk0YnVsZTFGSEF2OFlBelJpK01JTmtDVzl3b0ZaTUlLS1krRmJIczB3TmFiTWF4S0JZbXpmaCswZjJBUlZkY3ZMUmRaSENFdnBsMmhHc0hwZUxoRWN6eXg3dkxmZmd0WWowQzNtUHJQbGxIMkhrNTB0Um1YdzRGVEJQT0NVV2V2K2JCY2FHV1hnS3k1NG9tdFFKWEhWcDdKYjhad2o2Y0x4OTB1VFpFTjZ5Q1FsYnpGNG91OXh6bWZDb0lYdXV1V1NOM0NwKzE5cEp4RG1OM1ZiNlJ5NzE2VDJQSVBqZVpxNlo3Zm5RTVcyZUF5SG1uLzdKQVR6d1JZTGRGTzkzdWIvbzFzSVZwSXcvdzg3RXpXWFlSNUQ5ODNaOHZoaTh5R3U5d05HVW4zdzVCcEN6YkJzSlZlV3JOeDJ3N1hlUDNsc0I5cDJmN2IzYng3QkErYXk5Z3Yzdk1xOGo1bGIrRG5KaUozdkt0QlRQV1BjVnB6bXEwckZzdTRBai9JZFZJZlVBN2wwdytSazJNckc3d2RZc3lpWElOeXorNlBJQXUxUUpDQTRrU1kyT0V3RTJqdWYxYmRLQk81dkJGZXRuV2NzeVV4YUs2TWJqb0Qxajk1aE05d3g1QUcybThST0ppOHFhNmxOdkFnQ0E3enFmcVFDc0NwN1RnNE9CdFJqSUFmaU9iZWVlKzdNSml0WG1PVnRBaEpXSzF0akhIUHh5aEJ6RmVqS1NmN2hObWhoSy93cG1LZG8zTXNhSEVCSnRReXpZT0YxQjVwa0ZCSkw4WWlCa2NVT25HQUVJc1BBOWNGRDFpSlEydUdqczBhdTM4YUI0SGQ4S3BRRHdxSUZhS3dtdysyZjQ5Wks5TTFyZUc5ai9oOTcxR3U5Z1Rha3k3OFFtdi9vbU1ESlE2T0hKNitZWUo3TFJUQUJKRXNzVk1obFJSWEJHamJVU1VKdkFMWm5NTGg0VXo1aW9CR29jY0h3TGVyTWc1T05BaGZuTGhOWklZSzRXSnR5eGNQMFc0dkNMcUN1ZkNHeC92eEhnelZWMzErMHlqWlFCQlhDMTlralRqWTh4NDZaWlNVdW1SdkFvenNnWUYyakJtZjBVN01DQ1NRNFQ5ekU2QWxKb01lZi96dkdiUjVvSGtUM1drMTBIRkp1akt0dVAvcFBWZHRDTEd4cmlpREtobW1Uc2ozWjdjczd2OWFqL0NQSjVtOHY0SFoyQkVEZ3BkRDlkNU5GTlVUVzBROUxOL2w1emd0d01FRWdMSi9oZmZkUS9zTmo1SXlCY05waEtWM042bUxMN2UvUW1vYjVVa1FBUERMWSt5NDgxajdIMDIrdnZXeEhxUUJuWUJhNmJzS3JlcEZqanh2OEZqbVJRQURBdVIyVUFCRE9BY1RGOUxKUm5nMGhVamwvODZpUFZhbEpPN3dkQW8xNUlOMXVDd2tnUXdkZGxick95c3QxOTNndk56dWJNL3VERjNRNUVFSDBiTXNONDJtRzJhSDdPdHFCQkVJR0RjWXJGYndDV3YxakVmUFdNcVM4Q2NnS1o3Z1AwS09QVVVZNjFNY3NPZ0laWjYyNzQvVFB1S2dZYlNlVC9PYXZQV3djSS9BWFdWYXJtdXJ1ZzNpTXdOVmVJYmtQZGFwbFlaZ0FXN3ZyQTJuOHFHbG5CV3FPcXdJQmpKOFRjRlNtQmtoNWRVdHZoVU45aDc3OU5EZHZTQy9KTk1tNVlmNU1wamxGakQ3Z0h2d0ZndVB4eEhqdlZ4bnpST0F4RDliSVlock8vKzh3bFA4Ymt5OEpqVmg1Rm5tWEgzVmxTVlR4NGJPVFZHOHVuSmVkaFJCYnFBVmlkN01Dbko4YXR6THBZcFdVbnc2YWovKzRwZDdoSXhyRzYwbXVzcW1yT2pNTGh0eU9hQjlMMVdCcHY5NUtNNVhLR0w3OFJ1cTIvbWJ1RXEyY1lmU05SN2M0bEt5OGZWamlYNTlweEtYd2l1VEQwaURLbzY1Tk5BaS9jSHRrSWlvWFFDSEdVWkZXRG1oRHNNU3dZd2dWMjBsb3YxRVF6TGdJbGtBQUpVd0FBQk5sejdERmdiclRVVW9xTUZUU0FvSzJ6OXVaWElGZ1oyQUhBYnR2WW80VG5IeWNIdUVSblBVcDVUeVNpQUhIWnZqeUNXRzZiU0dNRUx5dEg0YzBDWHdFQnU1eEI3RVFBbFVpckd4MWhCenByb21MeThoN1RjaXVKTkJ4MGwxaTJRaW9meHp3SUNOM2NheGJFTjhBdVhSN0pvcWU3OVFESW9QTEtGNVNhMGpVdnkrTUJlSG5scTc4c3I0MTg4Q2hCQzlJNHMrTzRHbzlUM2hwSW1TMUJTbkhpenpseVVHVTFmNGZUR2E4bWhnZTJYY3lKdGk4N21EUVRpTU5qb0RYWFk5dVphRTgxR1lCZWh2eDRUdVArOWlBbnIwUkQ2Y1YzUjFvd2FHc0xUSjdic2JmcW94Vk5uNnlEUE1vbkV1dk5nbEU0RHVWOUc4UGRhdUxCQ1ZjUUVRSkN1S3dVZ21mODlxcWZ5UklITzc2dXNNaVRDdkVtejhBTUF1ajRzMnJHUndMcWNpVzZyL2tLdzYyWHNFWndURGlOejRHVVUyeVRMZ3B1eDZaL2YzcmJjWmE3a1pFQSt6K1RDYk9VYnNqdlhUZ29obDVuQWwrNmJ2VHl0VnBzekNFaGswK2ZYZzdlaTBmM3Z2cXJDYlF5WmRHSnhzbk1LTXZjU1BvRUQwalNWTThhN1FxdUdrWk1VTThDN29FQVZhaktQazJGcGFEdVFWaERDbmVDaXA3aXlkWHptNWxhcXBFdkZKbFlwKytMZVBKSitQRkIwTkZjWGFUVElhU3BUYUFYUkxORGJOMGxtWXBkbWt3bHA1eXUvcDVNeU9CaURnQXBHNk1lZU5yVC9heElCVWJ6OCtOTmF1b0FtM3AxemFsY1VPVElHSThHV0RQc3NXdzhkYUtvTUdmWjg5NmxRb0lHaG1NaFhoU0xrcVZ5MFBTOXlqMVZrZUVVdVlPcDlTbkVOeDBYOTBuZEZCdENEeXhsWmhwZllvVElsVkhUSFp3MDgyOEJ6UFRJeVBXY2RvRGZtSTBSbXdHWWJ1TExGZW1CNlhpU0x3Uk4wWkNnSk41ZDNrK1hMaXdMdEhIRUMvTW9iZlFFVmpZN2t4bDFsQldqOWt5WWpRZVlSKzhRRkhCU1NmNlIrVzdCNmlsT0VUVXJGNXlYYks2bkYvbnVOOTJrcEVRVFliREI4WkRIdEhtWEx3YWlNZkVETlVNV09xVzA1Mmd1UFVlK2F3ZlZOc1FSZ0pVbkprdzRBNkVseVh2SVFXSEQwTndKVnYyVmxKMDA0WFZsTVRweUlEWU1iaysyZDdQUjduV2EzV3BMQ292cURoalo2L012U2dLN2llQmczL3FqTzl5dVh4OFM1L2ptdWVZNEI3SHJ4UDhTQTBDUWxmbFVmUXdUUkN5aHVEMmhoTHFGZU4yTEFqR1RKNjZZTllDdjFpVzhCbGpRRjJ3eHpIMCsrU1BsZkl4NXdNQmhWYzBiUUxvNksrbkNldDNaZVJiR0MxdEtKT1VBdDRXdkZHSy9ha3Q3YW1FOFNlb3hwais0V2dGVFBTRFB4K2RmZUpMcGd5bFdkUGFMVHNLSFBtVitFTDlhdXJMT1ZmdmdsN2d5WTlpSHI1cmtIcWdNY2RST3hHdXphQ2k0aXNQYXlsSVp1cGZFdUFYTUN4WmE2KzBNQUYzRVpHMmU4K0NoVTE1eFNjN0I1Z3NBczV1MjgxbzVzUmNHOFFNMitWc1R0WFdRdE9aWWx4aXlMcUVBQUpXVFJCUGRRbDNFRW1lUWtGMHBsTHpJcWJWT3BSTVdudDhVRmRTR0FuOEQ4K0ZGSEtQM1VaZVFxVS9XMWtwamJyaTQwd0ZDTXRMOHFmRUUyQTlBTjBwTnZqcUFFd0t1cmFPQklvbkhIZ2hKQ3NPK1ErTGFud29ZQndtNC9ZRTdpMEpQL0NlK0wxeWlacDQ2ZHlMbzZlV3kyMVJLWXpzMmM3TDdDRlFteFJyOTgyUklLSEZyaTZYZUVCcmRuVUV5Y2ZVS1g5MzVET1J1Vi9MMnpjRWxpZUxzNytObU5LVVd6YitoN3Z0b2h0U3o3ODY3K3BHUmVrOU5yb3FJTC9HemN3NTcvL2ExT2pKclF1SFNmam5aZld4TG5CdlZmcGUvdXh4VlpKRi9WdWlYNXRYZ2VKMGJaZDBvNzhSQUJZRkU3K2p1RzVyN25DbmRKUUpYR0c3VFUrb2NNYndNeFRKbHdvMTZtVWw3aldlQUIrdnBScitVK3JlZEFFNGd1WjAxeTRuRjhQb0ppUzl2aWtBNXphSVp5bjhPWXBkODFWbmJ6MmVwbjd6MGwvS3JNbEwzUzFMNS9rc05NOStNdjF1OWVXbUVzSkxKNno4eG9MeWhXd0VKQXBTbGRPR1dYUU9YS01kSFR3dkhzTVBzd29zOWFyOWRuMGI2cktjQlBTRE1aaVRKZlVnd0E4ZDc5OW5qdGZMa1N4NXRSMnpEM3NTSmhGZS9yUG5lOEVaN3Y3SnhKNjM4cG44VWhJQWQ4T2M0Z1VKQUw0YURRdFpSbWs5TkhvY3owZXExV1c4eUVwZ2E3bkJTaWdSekRjdU9HTWFYelRVTGxzYit1L1pOR2RVbTFKT1dkRW9jSms2MVN1NklKZWx3STRFZmpYWHZMTG15N2MyTjNYbXN0WHVIcDBWWHV3U3VNZjhOZGxLUHZGcC9ucEF2Y1BBRGVqazdRZ0JUbnE4TzFQVHR6VFBFeFlEM1RKUXV5Sm4yYjhna3JHRWZRWHY1QkV2aDRQRERaaUpTTDZqYUZ0dlNIMlp0S0hvcSt6SkxJQXFKSE1KWmJ4M0QzSFo1VGhrM2xjRGEvdXg0M2dPVDRCekxrSUV1dDEvV1Jka1pLY3lBRFlBdGJ0SkMrR0VGUGY1Q0JBMkYwVzhmNXJzNWVHT1FNS0hNQmVHalppVzJkVDFrMWszWXBmSTAxOTZLZUZvZjRsbXpOMmt4bDMzNjNBTXZ1aVZKR0xGenN5bXhHR0JqVllHQmZhTE8wN0VZS050YzJJZENFZzRpSjN2dFVpQmcxaklVV3RWOHZrcDVXdFdubG1BcmRXeXZmQXdtWUhRVnQzTlROaUdPUS9Ld2hBZ0JKUXNaTGpRNU9IOW8zL2swOS9XejJTTHg1Vk5jdXhNZlpRN01IRnJpM1FGZTFBem5KZ1VobGIwczdrbmN6Vk1sYVJQSnVpTjdibzh6VklvNWlFQjJuaHc2MFdIRlBDOWpLRGRvNmgrekdKY29ENlhNajBFbVhqTXpyVDhpdnVMMWRQbEtkekRUUGVOSlRtMkNSQjkrcTlHMTk0Qjk0K3U0UDhrVE4vTkpNNGwwaW5McVowbVdLTzBSalBid0tBRXdOdHhualdYQTUzdC95b2VIWFFETWdCQ0JlUDR6Q3UxMnorSGlXbElJSTUxQnVnWVVOWkFXTlR6Vi91L29iNm9QaVVvUjNCLzJPeFFISnRyVTV2TTlTQUVZRnUxUSt0L2QxUmh0ZE9mUTZ0cng2enkyV0hoczl1WElZL3hudlE5ZnI2OWdZR1hwRmpCT05seTBlbG9kOTJsTEN0Y3N3T2RJckI3VTdUZ0hLSGgxRXNuMHBrbS9ybmZGcWZOWmUxaXhDNzNYUW1nV2FHcUU0dDRUK3Rwd2taNFFzM0xabUhHZEJIekhFUUxXOWlBNHQ2eHI5ZWJDTHhldW51M2hicHQ3RDVUQ0N0WmVhdEMzblExdFBsT2o5T1ZuWGNaczNEOTdyRGtwSHlyS0lDeWhNYXpqaDBXQmptbi9zOENCMzJ2YXk4cUhWeW1JK056TlYxYUg5SjhUZ3orTlU1M3dyZnNiUEd4OVBPejJpd3UvUDlyVENpZXFQSFlBbUdqOGNyMHg1TU5jUG1NWm1DWXRkSzhLamlPYUQ5MGVZY05VSWlsYmdZZEE2QStMQ3dRR3Z1NERUcXJGUy9HQjJ0cjQ3b0JKTk5lK2tnVmhxWS9IZ3lkTFVFaFVRQXFnL1BvbVNMWTBzd1oyanByeE55UW1DSWwwb2ZQaE1qRm56M3E5Z01mVGZONU9GcGFOK3NPUHd4T21LODR1SUdJbW9lRmwveTBjWnhWMjhXR1dHRksyZk0zc1JVeXBVZlliZzM3dXdOQTl3c1U2L2M5anpITi84N2ptMktxNUFHUVA1WmFzdHR4SEZ0TFlMaXdKdlB3NDl1Y3JOM0JpREdkaGk1L1dScXFBTkF6K0VIUzlIWTZmUmFwY3V6ai9BUnRQWDhEQnN4c3RTS1UzdC9HeE9BaEFiZHhjZnRhRjBnd2NoUFpWNmIxbXRWZnlJS2twamcrSm1DOVZvUVVRUWdUZ2pQN1kvVXF3bm03eWVyM2tpOUNlRHZSeU5rSEFkMytaV0RheXR6VTZKb2I1YjJHdzB4aDZTRjFMeDBpb1pVQit3OFhqM2d5WHlDR083UFdZa0lpbmxIbEFUNWZXemFvdXF0Q09iWjIveTRvVjR1THR0TkZ1VmxoNmdNWTkvT3hpK3ZsSDBtTC93elAvUG8yUWpUMzVhSjl4UDJHOGNlZWpKTmhPUC9KOGJiUmFUQ29HR2dVdlVzRHJucG1vYk4zcjVKRVlDOTdrTDAyTXNMcWY2QmliRjhieExDdDF5L0ZqRmhHRytPNXpSVkdqNndxemlRN2FYSGNldnNmUHpqSHN3YnZYM01qTUNZWC9yNkJiMWN6WUVpZWoyTy95VUdsUWVSYVpmTHN1cmdiSFo4a282eDhKVGd4a2pVM0lZblRTa2NoT01mQjlDTGNIS0xGQWRRUUpHbkgvWkNwdEx5a2ZqdUVZclJPdFlnL3VuMFdCeXZ5bEw5ZE55blEzWGRUQW1QWmtmYklWYmVHYldEakpvOTdCdWdLb2NmUy9aUXp1VEtZL1JtZzZpQU4zSlNENGs5STlhSGxReHNOSkZzV2hKQ3Y5MmU2emxNbGRGakVIM0UrWTc1Z1BMOVpoU2hFODRlck1qSHpOVE0rK25SVWEybzZZWkNMSXR5Wk9GWVNhQlR5RkhmOTNYYlo5YnhuTjE1aEhncTkzaDFmcGdrZ0o1a1VjZndiMi9CUFRaRGVEd2JEeW1SeTkrbFRWdDhxcTYvQ3o0ODdoYUFXRnNlbis5MUFZNHZqSjF2ekt1NlJHYmFPUEtxdU9yTFYvYi9vdnhkNkQ5ZDNuMjM1UUk1NVJJbkZlTnNIQzBVSjNZRlYxWnUzNmQ2WDc5bkdxa3Y3eitScFZaWFFyK3BwR1hBY2dEb2phMFJHUDY3RjZ4MC81MXlmcDJ5ZGlJS3JnTUFQMFlQYUNBSExjQ0VaVHZPVGljUXJZRFhHWHpzai9sQkFCUkNLRGRxMzJQQ011ak1PekdlcENqL0pvdnJsRW9NWXA3MXgydVVpNEVNUVNZUnpBdjUzRkNTTFM5VXB4cllvRjRRVkhranRUUy9KWll1Slk3bTVtOVhCVDEvVlJFSGJ0ODNjaGt3Y0QwdkUwZXh5VUgvN0hkMHlaZStqcXpjUk84MVZoSVVBQkZCSVR3bEFZUUZOZmY1OGp0MUllRmZjZzZyVHE0S3ZNbUo4N0toRmdHcnBrTzRadHk3SUZCa0RNZitlWmNidGE0SUc0NUVXM2QzYm43VE4wc0RNUmtoQUE1c2dqZzJsQzd5T0FrQUVDamp1YXVGK3F5VE5qR0tLNVhDVkFVbTgwU3lsL1lLajlqZHlVbitmbjVUY3lMdHRwcGFFN1ZRY1hXL0VBTVJIZ0tBS1NLNWV2Q1BVN2RVS0ZzYm04SUQ4YU93bnNJUFJta1J4UkE1RFRFMlhocXR3TkRabHpJWWtIMVJidFFyY2FYbERGaE9MbTFjYWRNMllLMzlXYnQ1VnBtcjY2M3ppK2ltSC8wNjZTUVJvaGxaY1hlQ2crN3J6Tmpvcno3TkQwb0ZjSEdoYTFDb0QvYmgvYTBvL25naWdnQ2xrcWorTlRjYnNMTVlxaEc3RGYvcitMeVN4VDdjeUFPTWt6YjZ2RWYxbVhlYnJ4MTFvdkxoQ29LNmVnRmxsRHlsdWNVWm1UYnYzeVhJY3pHT3ZodTdVaXFwbVRHUmp5elExbVJVUXE1anBvUElhMFQyUTcyZXFxY2Z1aHg5WmF4ZHpEd1MyZkMvblM4a29mbjhaV3hWVEdqMWpGbXUwMWU4RjJGOVBXalRiQ1BMa3RVYVNYVlNzTW43NzZ4NkF4OFp6QTBER2lFL3Azb2JFaldlR3Q4aHpSOEg3UjhOVEgweGljdFNRYTB6Q3JZNFlKSk15SkwraXEzSld2TTZVNUhmMWdVQ2diMnlKcjlkY2NiLzlyUndvQjhOaUVwcXdhTUFPRlJ4VFF2YkxUektxUWdPTWdEbElvd0VLeGZ6Mko1UE9JQk5xRy8xOVQrc1JuTzZrUmdOcUJZWUtNUUdKVVQ5MzNBSWl2ZmRWVWUxK2owZzU2Vy9hZmpPcjZwWHFkSFc5QlFPR2k3RnBkR0FnN1dUTnJneXZlZHR4UXRhb3I1OXlWeGxTSG9JZjIxWEhKclJqOE16SE1xMzNvM1RRQVpoRFg3SS9yZVZ0clp3ZytIOHh2OTlxVTQ2U2R0T1RtMDdxM2p5OSs1dU1FUUdnYmovTmxqbFRUQmprQSs2WFFuNCtGTzFPeTZWTk9nN2tXTVZIMkcyWWEvcldyK3kvWHNyRndiUHFiNm95bmJwYWo1YWI0a2xRQ0dGSlVUVmVBaUxLRk9adk8vVTNkcENqQXdLQ3ZpMEdaN2F1UTZJWm56bVFDVUxYcnlLT3NrY3JNVFNFc0dDTzZHa3NGZXl6TEYwb3hIQWN6VzlwRUxRNUNyYXZCUCtWK3VYbVh1M21LRXlEYWE3TFpySXNuZDBlY285RUtnQ25WKzBWazNQcC9aa2dEZXQzSVE2dXVFdmh1M21ZWlBFS3A4dnBSbGdSK0lSREdmQWFEVmtkanJmTHlVNGhReUxQS2dhaVRsOXNQUXBqZ2N1QnlZYWdvZEsxN2JlYkdPV1lIVlYzRzJwSWxyNmZ0VXNaYWV3RGRoOEl3OTRXWFRJZEN5aDFDZUgxUDFpMFJ4cHorY1MwczBHUWwwZkQ3cGM0b052SjRHdk1QMGNWWTBRSmlGVGpQNDBVUDEzZ01Lc2Facmx2c2tRN3c2YStYNUIvcDc4ODFwVGR1WldyZGRGL2MvdkdnZXZ0TkpwdThHTTF2cWludk9RV2dFZEp1ckwyVStQM016N1ZuSUNRbzRxLzFZR2F4TUVSaEZQMEhaTG1LbWhzd094TXVQUW15eWZDSjhYR3RuVFdCcU5CYThRSldpQkJQZjU0c3VLcmZlTUZZOE1sMnQ2bVgvOElKWFBYRzVRTFRwUHViSTU0V2RiUmNpRGcwL296TXRRalczVnZnYXNHVkVIa1dkOGR4bjFhL0gyTWloemc3SXpObVM5dHFvVzR5T0lhNmVHZ1J0bG1VWTRqNXBxNWNHcldta3NET0VUUENkMFovck5KVE9XRmI4KzJYYldIVERQN3VVaXJ4VVJZZXVzNE4yZTBNb0xLeWpRcS9pQ0xBTGlZeWwweFNiZWZmWlEyczhNNWFHdFYwclBubnZ3M0hLeDRoNUlucndPS2drY1FSa1NkM0NqZXN3U3BVR053bGpxU2lQeWNlVlJaL2ZXN0tmbXpOWmt3Y0N5akdrMHdqRENhZ0xaUVdFN0k2MFJMSEVDQjhvdDVadHBoYSs3R004NG16dDNSZFJERDJNVVBoZHVPWnh4UmN6UnNYSHdyMjd1ODVOM0NtL1FtSmVNWi9UbVZEUmd2cFF2VzhsdEtXaVJRUldQbHYxWjdvWjFuUDU1Q0pmOWtrRk9JUU85dmlXbDZlWVFMQXlZSG5rNTRoT0xtc3JCVEM0TTF3enU0NU9GYlVtK1o3L3RLckZWNmg1cng1U1g3RmlaVlIyQnlackRsTEUyNU80Tjk0V1hrbVorVk1zTXA2YmtPaGtEcGhZYVNESTRRYnhKRlBxK1pTVE8rdWRab3ZIVXNPNU1XTlBacEc1ZWkvbllLa0ZJVWlGdURQNy9CTHJNaksvODAxTzhOdzhtWUNjYlRCWTNGTnpkNWZPMGdtVWQ3aUJFUTM3Mi9qTEFrRUR4MWRTNFZ4RUNBUVhmWUdMTEorYkxRaGpyR1IzM2c1elBhUHo0KzNuYjFuWFEvRmJWelJBbmxHVlliNkNVb08rL0VJUmo2ZjhvVUY3TzdFajMxZXpqRkFiRkVOUnNEU3k0UkZQc2NYZStnanV4ZHBxQXNLWGI5elFqd2lZMnZMUDFmRW40eUMvYmdpQWhDWWpBOStTTmNhWE1FSm4ycXNqTmZkK3Z1WTJPV0NqK3d2UHR5b0kwM28yMjg2ci9adkZTamlkaStuV0plbGkxcTJIYmNJV1B4MHBrWDNxNjhGeGU3Y2JLUDZMcDg2U1RHbDBiYytLR2czYjBaUWJRMTZvN0I4Zk9ZV21OdXdnUDc2UHY3eFIvcis5aVZYWjF6OHJjL2NrTmppSklvbmVyY2UyQlNoUGFscXhkZ211VC9ucGE0UlMvcGFka3hpT1JWTXJPcGZrdDIzcEdySmlUOEtVRGwyMFhLcWFkbDRaczdxTmZ5Z21jY0Rmczg2ZlNPaDJCM3J2YW9VenB1S3d6SHVpb25GdmtsQlBtdDRlYzRGK1ZkT0ZrVUtJK2ZEMktDcDVPUkxZS3BVRWd4UkhJcGZUZSticjNWQ1FaOTQ4aVJ6WlNlUVBZc3UvYlRZMnp5MzVwcDg3WFBaWXEvV1M4NTNWd1VTalJJYkRHaUk0bHdBcy81YlI4NDdKekg0eEovVDhjb0JrR0c2SVRaaFQ5Y0VKc3RheUxUeFFFZ21ac1FyaEgzdk01eloxRUFjYTlEbW5TUEJBV0xIM0ZqdkNFODNLMXVNajhIekU5SjN6SGtveW5lZVpHc203RjNWU3dEd2htek9jdjdFMDVsZ0taVmllR1JxU28vY1VCOC81VVphbmM1dDNabks3S1hzNm9XUmZSSVR5cWpBd0pPT0U1RndweUNtaDZyb1BYbXJLc1NUQStDZzA1dXNYeWpjclZDb0FTaFRWSGVsVTZBZWtEUTFwaHZrZGJGZS9xb3ljTjVXMTdmaHUvOGNrdGdpM0Q3d1dieUhvdXhkUEZZRjNlUUJNVkdFUVVpRWNUREQxTTk1aGxQRlpZWEwvRDNRTTBweFZnYUg0bmp1TWVGWWN6YUZzdjR6K0x2dzFTejZYcEMvZGJoNVp6YTNnVXhZZUZHVkdFamtaU3I1RlVhZVJJWVI3UVMzbDJ5R3p5dkhUYTFvMFh4QmNEWHI2RERPUmlCSW1xOEFBd3FQMS9jOFRhNXdUdTRBMlE4L2RodEhSZXRtaDh6Q0NHWnl3STlmOFd1VnM0QlRidTdXSnpobHFMdm1hMS9wNEFGaVpaUCs0d1U1Mm9NYmo0L3RRS2lWL3JBMXVCQmI5bnJqaXRMdGRBKzZLYTBXWWFhd09NRmljZUxRdEZTN3JKMUh0QzFad1ZDNXpveDYyNmVLKzJxZjFnc05taytRSVRqUjlzN1FlSzFJdzQ2ai9WbVVsR0huQ1BzZmR0ZUtvdGY5Mk9sbk1NR09HcWg1L3I1RU9KYnJVMkk5eDJ3MzZMRVdYcjNPVkpTbWtRSnZ2anRteDVoNGlkTFBzdHJZM2FWemk4MzJwaElXdTV6L05VU2hKbmtzSkc5WmNRWVNtRGpXZEdzcHRQWHYyTk1CanBLYXo3eUtMS0ZtazA0UHZPc2Rid0dqdmRlWUJabXhFc2F3RHA3Z1VOMnorK1ZNWXZWODd3NFlCRUR3T0ZiUlJoS1ZxSjd6UVZUS3BKL0pBM2RXWFJ3R2o1Q3JnbmVIenltdGFBdU9iOEJkT2g5cFpUaUFxSlE5djdmNXlhL2wwNnVMbEtSUHE0Mys0eEhUd0NsQXh0amV5MnJuUjJDQ1l4Ymt5T2FqcU9qSVdYVW1RZWMyZTViM2Y5Wjg2ZEZCN3lJUWhtU1hiSmtRYkk0Rlc4Zk9PbmVKU0w4UkhFdlNjNE10ZGE5cHBJc1pLbkxOM0ZML3p4R1Fuc2ZRSTlSakY3MXVYK25BeVNDaXRwS0hleEU4bmFSUkRWWlNEOXY1TWRPM21JRXpOdDFsRzBYMG5CUlBNcFN3czd1eVdVcDdINGxxN1llR3FxSXRqZ3FQZzZxK09palBnbmxEOGppSS8rdWlCYkVLWHRSdmk1R1ZYU0FXdnlrb0ZyT3NBZkM2bm5DdFh5TTMvdVJ5SkU0eEhwaWRCSnh6aHA0a0RTUXZYWE55OGRJSnVVc1dNVTQ3RVVPZ21BaGhGRlVPS3lXNklwK1pzU1Rvc0UwWE9CbVdlaGlwMUU2bkdtdWlnZ0t3a3BKeWNnaEJGejBtd3Y5ektqNjBqNHlrTDlIMXhnZzE0Z1NYaE1pYUhSQ2xWNXc2RXdsaUNLTVgvZlZ6bmo0V25GdjlVQkN4VkV1RnZQRDc3UmoyMGFEZm5OaTJZRlJFTEN0c2ZJZVVmREJEN1FlSnoyTnVHSXVKZ1l6NjczMVY5Q2RoeU00WG5MYWNlTTlkOWtqSUlqc1ZFbzBncUFHMW8rU0I2TkhvcnpmV0RVTk1OVURKRmltQTA2Y1BmMkxhbEc2VkhKMnBGc1FMMHBhbHhsQ29mVVVkMnFNcjMwd3diSXoydW1XQkxlNW0yMFZreHpLc3FQa3FyOEZBdWU4anh3REx3dmluM3hLM0tQN3ozZ01SWXpTbTZKckVnRFphVnl1RzhjYVVwWkxVSUF5b0licHNHRFNaVzA0eEZ6NWI1cmpNYTRKU2tpSGF6dW10WWh6TkN5alJDdngxWmE3SHVwYVl1S1Fabk92NlhpZ2VKaTE0cHBBdEpJMktLdGM5aldqdmlPeC9CR09ZVFBZWlk1bjB5OGN5MENzYzNHOXdWbzZWR3NkcWhBbHJqejk5ZWZYMjFpbVR4WFhqRlBVUUVGUzVFOGRWTDQ1ZDUwc0NNUkMzZC9IL2dJTzhHb0hIbXpmVC9RS1BibXhSU0hJaEpBRG9QTk9kOEtUd2RaeUNsVmRvM2VzOWFEOWtsOFZOTllFR3NFaE84bE9VeTBhUEkwQmtTOTQvVGtiVFhham5oMjA0R0REeHZrZWRBMy90L28wMno4aUY5TXduelBvSkxNT3FMd1grRXZGYWpWMy84NTF1WTZjdUU5UDN5TE8xZGRycFZTc2IxQXNKVVMvdUNUclgydU9ibUdRYy9VWFh3MTNnaGV4UTRZcTI1enVLL1RYZ3Mva0o1aG4wMUhtUGIzamkvSW4vbkl3QzhTVE5zY3RxMmNZZEllakdzMG9EY29IS2J3T2s4MWVaQzk4NGVFVXIzSkFiZTY4Nzc4enM5N3RFK2o0Tmg2MUNkUGNwb0VYN3A5RW1pTTJ0VDFrOTlzeHZjS3Y2ODdHcTFkRStZa05YZGtBbndVaUJzVzZLdHg5djJOSzJFMExzWFc0dFU0ekV6ZHo1aTczc29KdHRYa3lIWHNHR2ZqN2svOHkvV0FmZDNzb2RrMGl5UC9IL0ZMb1U4c0VHZm1uQkJHdk9Cc2w3U0pCQjJPSUcyNkZLMExuYWxMSkc0NTVkeG1qZWRMckVyY1J4djlmem95M0NxT3pKRmx3Rk82ODFUeHI0bjByY3QxWTh6VTdmdXVpaFRIdzFwck5DNEcycm1EMWpYZGJhRmQvZDJMMWdSWGJtTDBqd2h1WVdVbnREMHR3U3pZSWUwV21mcGN0Zk1JQjRlNHpadHRsZ0J0VmsvajZMVElPYkphNyswelJwYzZPL2hBN3hHYktXWFlXcklxMGNaMlV6K3gxcmFxdnVudkZ3cXNpOGc5bVp5SEE0bnJoeDV1aW9XbGFwb1lTWitEZ0d4bUVwQk0yUi9GZmd3U0g3NDJ0TTg5NFZ1c2t2SDRjeDE4cU5idFYvK0IwdVZlem9mS05yZyt4RTVlNkdjdElsVURYdnY3bVRyQm1tdEFieDhCblpRdGE2RzRlNzA5UG9wclY1cUpvUkNiRko1cmd6aDJPSFlteVdOSXprN1JpT0tuaEJTanQyckgxY2ZOclNxVDRaNy9MRGZIc3dlWS94N2xxQTJMb3RVU3dBaDdsUzFvZTI2eFpvTWJuS0JlK2RIeERrNmZnREV1dXRjZGp4ZnRoK3RsblFDbCtabTQvOUR4SEZrUlFtZ1FzQit2c1JJYVNPdEMxOGtTcDB5NVNDeGVtQkIrOFhJYzA2TDU5eHRnSy9TTjhKTlc4UlVDT0RBd0FHdzJ3ejcwVGZHTVUxRENZcmFnU25UWW1ld2xDUGQxTm1OdEQ2WEV6eHdVR1UxSlM5M3BUS2F3REN2ZS9XdGoxT0xWUHU3ancrVWplaEtqMDFwQllxYThlTGxBcE9jTkoxaU5rbjB2T2dYSlZmRUt5dXgxdk83d3NaVjA5UWpSeVFZa2FJL2pJSjd0eDZkMDVQZFk1SjVGdHgyZmNMTkJZUjVDSUswUXRXOTJ3SmEvUnY0Vkczc2ZCK053WUtCZ2FjU3BhU3FCamJ6eUxWTW1EY2R1K0ZCY3N1TjVkMWRDU3N2SzdYWDcxaXFCVnVlbFcrZW5VL2R6S09SWG80bmJZUERyUVR3TDNaYnd1MGcwL1puWlZibWkzSnQxV0xzNm9CbkZDcFdJNEJITUMzNnpZMlZ6QTFHaEVLSGJiTWl1NmN4bXJZcFB1UjlpNHU3M0lnWnBORXZwaTlrRGZvQ1hKSXgzZGh0TnBBRkNLWXF4bGVwT2xzUUVDTk45MUN2UlgxUklrT0V2YWtRTzhVRzVYbkR3VUllRUtlcHIrS29hVDQzT0swTURacWx5b2w2NmRIY2MzRkNJcU1YMnFESUlqMFB1aVFyZVd4S0ZQdkZyWmZoUmFsOVV6OWxSZGVtMFRiV3JqbmllNm1Wd29KODkxVlJkVG9BUVhkQUFCNXp4amxnNVUvZkJ3OWZTcVhFQ2t6eGN4eFBHZjVta1YvbVdObnJZMWJQMTc5K29hRmg3UjcyTUp5UjFkeXpmVk8wT2tNM3lqeVhTMkY1a0hXT3FIQkkxOENDWUQweWR0a1FuQlJ0aHlrWlhvWXRQb1BpN25EUlJrV3p1T2U4anBiYWhpMC8rb3YwazErbG5xVEpiWU1sNWgzQ1BMd0gzQWFIL2syRktFRlptTkFqQndyVDdBL3ZUNUlVclVzd1NIZWcvbGZ6YXpDU1REc2o1My9JdGVQbnVWK1dYa0N4b1NJNHo4a1dHOXkrNkw4R09EeElPdlBRZ0toZ3hIQUNkck1LVmhmYjY2eDBNNmVQWUp4WjNVTE9jY0M0c3V6UGNnOWdzZ3kxMkw5SmFJc0Q5LzZ1dk1IZ0YyUjhvejI3RjRTUk16RHJ6bnk5VWRTR2liYXdpZTMzaVhzNjk0Mk9JeWorMXVteDRLZjE0bjNuZncweDR2ZVpsNjVOTk1nblYxZm83WmQrS3hjOXMyc1pBblQ4cVNvVXBTbnlZbVpsRTZnUU9KVFpFd0E5czdoRHlmOTRiOVRoOURYNnpSck10U05US0hBVXJILzdwZWtCRFAyMkdjalcrWEdJQ1FhWmtETmlVMFc3WjA0N0tXYmdrVzhRUHRvNG1JK2w1SG5WNVlhcTY5YzFGRVl0cHExQWRZZVhYeEFFM0hJVjdGdDQ2ellZNzdES1dzK0xmeW4xalRIdHpubjlhVTg5dHpuUHQyeDRMQ0pQTHNRcm11NEtURFpkZTNGc215a095eG03QlMrT3FGcVNnWWl5WC85ZUpDRWRNaWk0Skhvc3hhc01VY1lyVm9SZ0d4YlU1SkFzdjFLdUZYcUxkNUVZekZPSmRDWGVXV3dLVXdtZ2pTK0ZyY2l4NmFCZzI0d0x6NXFiRGF3Rlp2TTRHWUEvNlhyRGV4N05nN280b2RuMkYwSzJTUGQwTUZpMllqcHRENkpjTkpBdFFWZ01IY3kyT3ZKcjF2dmo2dGhWL0hpNks1bGQ2QzVMRVAwck1zWU9UWUxmNU9oUjRxN21JUlRKUkZ2QVNRRVYydUxnQ0pZUldvTFFjeGw2bko0M1RMNHh4RFlJQmRLak14OVhNM1EvVU5QUERkYjcvQzJqQWh2NTJNU3BadjJaV3F1cDQ0bHoyM3g1dGwrM3R3cW1XMC8yQVBPMlpRZE8xQzBQOWFjTksvRmJEYnhhc0lOa3haY3hwdE1vU1EySUFSRWJTUUozKzEvdW9iSzlxREVHQ3hWZWdlN04xenpkVmMya21rbW5jLzhwNUk5YTBkcmJJZkJkTXV5emdOdDI1am5ib1o5bVRha1VzMEhIRnRvYm9lUmdCU05aTE96MVU5SW9qaVp5SlZMWnZKd0JqQVIwQ1ZCMFpILzZyNW4xZHZuRXUvSGxIU3FCWDZGZm41akxNZVJoTy8rVlcvQVNSVXhCa3M3eEFPemU4TzFsTHJRYU85Ym5ubVUvdlNNZjFTRHJ3T1JWSHhER1FIdk9aeS8xU2JzK0h6Y215VWhDbXJLcjIxMHdRcThNd3Z5VFNCVTNuOTRIUFo0MG9sZVh1blZGYVBINGplSWdHd08wVzdXVGxiR1dwR2I0OGU4c1BxaXcyekY2SnpjR1dERGMrVHRKZHh2dmYxcTZwb0FDN3ZnSTh1R05LTkh6NnZkVVIxRlh5Nk5yRG1IM3RDekxrdzZhVW5MMXhLNkRQajRnTmJtSnB1SWxNTXl6Zk9US0Nodnd0RHBHNnNRNXBKL2doZWFxMjJTTTlVcEhrL0lYN3hZQlhBaHFPSUYzQWxUQnkzOW9DSThVbWh2ajJrOE1wc3N0NWZ2cXRpeENxZFRoell1bjd0NVZHa2wxZHlZbm5MVkVsNjVqNEZxNHVlTjMwK0E0SEJmRzJxN0xXazdYMENtMjRWN0hnZVJqTGo0MEYzQ3MyNnVpL1FzQXEzOXkzU1lVNVppUzgrRUEwbXUrZVo3eDJDZVRRbzYxclp1ZWhmY05UTXI2YlZubHhuYVFzbkVIN0hVL2hRYytrV1FaS3NkRExtZkpEdmM4WGRUZ0JxSDZJUytRSitCMlluVnc1cHlHT2c5c1FPRmMyaEVXWndzRjdGOEhWTFNrUnZLcEhwc3BmcjBDVXgxbHZicTlOK0VoRmpuZlpxMTVrQTJSU1ZZengxNXd5TXlacE4vK1NXYXN5K3lZOXZpdjVJYjMyQ1FzRFVMNis1YVpmVHpabGwvN3NtYjd1UUp2M3NReE9UYXNmcGhUbHRYOVdLWHlOTnlJcWl1MDZzTFBRVjJvT3VoeEcyYjNXOVUwYVBrd0RNZDVVczBJMW1zUURBaUM5SFB5SCtnbjZ4S0JLaEJTQ0MwczlkWmRkMVhmVGpuWTB2aG5DaWRrbDMraWtKYTBkN3Z5R1pQNWFrMlNyWWJaRW5NclgvaVBIWG10SllIa0pMa2JBd01PZ0YrZXByODNuOXhwWWhqQ1JnS2xNbEFHRWdrNENLRW9vdkZBUzdGUytTUHBZajc2RC9rWUo0NVZmOWwzTGdNNUwrelBaVG5ucnlHZGg2VFRHVXdrYXJLK3pGU0k4VlJSN0FBZkE3eC9LY2xNWGVwK3ZCdGdaTitkQys1MGJlUXlKazZSQ0h5S2lrclBKS281TGpsaXp4dEtWQkY5RkFZRkdoWEx3YlVHRnpJWEdwbmM0R3F5clA0dG1IcnlNMkdkQ29WOXhVdzh1VW5pTVMwcW13Y3A2dVNtSzBRSnRzYkgxN2dRNGtwMHo0aVVJVktObHdCYXJLb1kwK3VRcENjcGxqY0FEVDJ3MzVzemFwNWh3QVkyUmRPZmlvWTJOVjZkbVlpSTdTOVZJdjltVzdZNjJoMGxtR245dU54ZXp5dFE1eGlwaHpaV2xUNjZORFk2bmF3aXVmMjVvT3R0QkNLTGw5RHNWYTVQekRVTDFZYVB4RkFubFpyRXN0M1dkcDRqK2NVRTFjN3plRktZWm5ieXRpdVI3TzE3RDFVRGlpcm1KRDZmYURWd04wcTBXZWhBZk5Sckd6YU03SXFVOHdSQTk1ejI3NWpFQVRFL1ZkYkkwRGRCTnVacXBuVGt2ZmJnayttNW05MlVFMTNSUEZDcVRvcjhRYmQwZE1QVDE5cTZqMFp0NUFOeit5NTFVMDdPcE5QNzJxK0RBR0NhYmd2VVNUQWdHbEdLampza0VNRFZ5V3k3ZGU0NldPWW45d1dmQjNObWVxTGdxcGgvWm5mS3g4RXlxV0hBem5BbDFCZzFocDR6NjFpQVFEdGZJdkNYL29DTTMzS21SWVVNMExQUjRXdU5aVWNpVFRhRlFOcUNMME9YcW1JMUxUM2loUlJXWTRTUWpicEJJUm52S2RrN1pnNUF1RU5qenpmeVJZWnA0YUdyVFZXaG5tUk4rK1JBcG1lV1dBdDM1THNqZ3REeFpqSHB4ZEUvb2FJN1dmOHY1UWNWT0JnakgzWU9FS205dkZvTTFDUCtZdFAvRCt2WS8rbEVHSHp5SXpEazdrM2Q5L0MxdmsvQ1pMYnlrQWNORWdWZ0lBUkFrUXQzZk1QYzJnYWpXMzh0VktOekYrZ1JzQi8xcVJkdWp0WEpXSE13S0RLUFdaVlFRYWp6NllWQktDZGJxdzdaUWovK01zQ3hKYkxncVQ4NEh1M2V4VllCNGlzUHZMdnpJZnM4NTdBTmhlRDV6TmJucmRIZno4a2YwWm14RytlMURGM3Vsc093UHVxbTZOYzJMTFo1VmRsWUllVHhpc3FpRGdSdzY3MjdzZzhmU2ZUckZuK0kzUDc1OTVNdnIvRmJuQ1BPY2hObGR1bjJVUmJzSnFySk4rMVRXNDhWTmlzcHRPcHVHbkU1U1dYSjF1ZmR0VnRPVEk2SVRYODVKcUlJanBCRUdydWt0MyszQVBvTVpsdjRzTHdzVTdYNWJwQWdIVWJzRUlVWFBjalNlL1JUaENpL0xBb28wWmlHK1NPdVRVWUNvdUxFdUZzbmlxSkt5elA0eHMvclFEYmJESnVub3ZKaUpUU3dndXpueHRvOVZ0UWIrOWxFQmp0M3BQdmZzNjUydVBWU056MXZBckg4bUNTU1VyQzNFN2R1WExqYkNndmcraVd2VzZBRUgxa3NJZ1g5c3hwYUFSTTEvS09xMXRTVHp0Y21yN1FiOGJydEEySGt1LzZKOHpVZ2pkQlc0NnA4dmFMc1FPVFk3ZXU4OWMrUzJKdkdFMnlzQ0duT1B0ZkR6ZDlFR1ZkMXBJQ3lDUk9JL0FVcUlmclVZZ0RYd0xHSjdKTk5keUFGVmRkeTU4NVhyR2tyejhSeGlqdCtqUzQ2M25QWitjTTVqOS9uQUtkMHY3Y0p1Z3k0QWtvb1FUWkhmMTRSemJwWDUrNTZvTWsrVCtpdktMMnVrK2JReFZsTWlDREFTYkx6VUNldTZKM1Q2eVNJNGZtV0hBYmNMdS9nSi92NkpMdkVwcmtOLzJFU1pYVnpOeTNWZG9UM1RYejZiYU1wdlhWZWNLWTRkeFlWMHZGZElXOFJ1N3lLVkhJR09PQUVnbDhsc09weC8rcWZlUHRBVVM4WWhTVlY0WFFpK09ZRHE0a2FHUXFHTktBSWlrVDBjVnU2TWFsaUVGL3Z4NnN4RUNkbTlEdWVoeWJwenlHSEtYVDFsRms1ZHFvOW1ZQTR1WTBiZHZlWjE5MkJaU040M2doeG5Fa2t5MUhwRnMrYjlacGx4eXdSVVFnOG5uQTU4aHkrWEZFaEFDc1Ftb3JybkhzekdRUkdCcGhEd21Sb1lUQjNWK212ZHNEYk4xdGJBRjloMmI2Yk5JRmxFdnBYTHN2QW1RT3JsVVdzZHlKZmZzMFNQUmdmR1l1c3BLMkV4bFZ1clhFSWdoUGlzcTJWdUNTMzhHVmZkUEtteEJKaDdkcmQ1eWl1enB0ZUcxT1RjNFptbGRlWTdDZWFubi9tbThUc0RHNkNYNVZSVVUvc1ArMVdPeHJCbWIxZXVwOUFwVWFDMndjekc2NVVSYzM0c0ZOUkc5ODFEZDZGd2EyK3RINk1acjVGNzF5UzhCYkU5YVd2OUtaOTNncWg2dEhQcU5wQjcrWHdZblZPM1BmMG9meFhGYng3dFJpRGw2QjFvbEJGOC9jaVQwdlorUWlzblZCdGFMekVyUXlFdGhPV29JSnB0SjZCdXhMcnBnYi9LQ0p2VHZGSk4rbE1lcTlRb1FKM25ibjVGN2FOZmVaTzdHN1NTN2RuRVgzVG9QUGxZVlgzWVpnWlgvYUlvNWVsWWk3MUNQSExNRWVGY1E0b3BRWFhsMjZ6Zk9JaW4rcFNJSDJ4V1l5UW1rTUVIYTBITTJpc2hjVUs2TTVTaVE5aSt1RkFBeXdvWDNOZys1WW9OM1pJK1U2c2JCZS8xd3g4c1A1WGxnQ2drUzAyMTJaa1EwclVJaFJrS3o5SHEzRjVNcGlITDlPUkl4anZIZ2hISWNVSmM3ZWFCTWlNRVkvcWt4QzNQbWh3S3lVQ0ZYMG9FSGQxZUlmZHpnYTNlRFYySWIrY01pUzV0WTlXNG5pakZJSGtwOWRidU9paGdicm02VkI4Vm9LZjhtQlJ6cTB6TDloZDBzV0J0bDJ0dFlrYnZQMUFRbkNxdFl3TVVsTm1aSlJGbzAwRm1WSUdDRTFQRzRHUU4zc1RtckF1alA1MGMzR2MvRVBBMVN6K1BWS2dBYWRCdTRUV0RBbjZOeUNzUFVJVUJlSlVBZ0FDRVFxejFxSlJ4QVU1TWxCSVJKZ2JrSHkzSHRpUDRkN0p0SGIyVnNYQ0ZBSklGWW5iaG1kemFQeXA5emg2T3ZtMmE1TUh3b291amtRc0JCbEFmVDg5SjNFSmw0U0RxVDNJRUV0SVRBcXFHQWdzcSs1UndiVXRJdVFyMXEyNWFQL2NuV1RlOXRrNWRjMTB5V2Nrejg4RE9Jb2FBRHVLT2JlTEkvL2RjMHEzQk5CVUhyMGFuNUNmVEw1RWtmdk55OW1SUmpzT09yclpwR1FjSUZNN3BpR3FKbTVLV2JxakgydUhRaUFBSlg1QnhtU3ZKWVl5YTRMeStwWEhEWldSK1VUK2F1bjVpSHRJakxOMWozWHpqdFFZVVh1a0VpOFFLSjdBb3FzdVhlS282S3cxa2lka2FpY0hCVnlVc2lNWjF2b0NPNG9HMVdGRVp2aUNQempQZ1czeS9uYVpFMTM5OTUydXpucE5DSUE5ZWYzbWtka2VDSW8xODd3TWZlK3ZKWlFvOVJEZ011a2ppaU9BN21Rd1FScUJaeFhOUFVDRWJ4aFhwRWJ1ZlVBZU1rQXlOY3lsVFA0cTBjaWpDaXdieDlRTWlpM25yWFdKT0JBdUhoR0tvZTJuY3VpYUFNUVNJQmtyQ0NRQ1FoQUZ0b0dvdkpoQmNyMmd5WExqNDRNZStOOERBdUJjamFCT0I2aWsya0dXaWpWUDR2SWo3WGc3cDdCMTl1b0tFU2RTYkpKT2pYU0xuY3NDNURkRkh3TVRaelc0MkVpeHhDbmlHaEI5ZTFNUjRwNmJYY1ZXUG1kcEJHSCtrRVhrZEsrOEplOTF0c1RVUFdLMXpmT3JremV0dnhRQlhpRzJ6VzlFUjF2RnlEOEt5VkZUOTlkU1lFaWpLSlZFUXZ6bHE0b0NmQXVjU1BSRkhtOTA3bnVsUnQ0eWZFWWpGRW9Ib0VsaWwvbVVBOHphaHc0dm1HWHpjM2tHcUx2MFFkUzJNYWdRQnBiM0FBWFhMQTB1QlA0dGVYeU1nank4MnVZTkhUMWZ4MmlwRzlxY09MZHRSaG1IMDhFbk5jcC9aL0x0L2ZyM3phYjJaT0xzZVFMYWV3RnZ5Ty9YWXl5K1YxQ3gva2UyY2ROYnlaWGsxeXUzTGVGODBpa2FCMXN4S0VlMzdsZjRvd3lqWmZycnBSbzdKdFRWdzlBNjF4cG9vSHA0UXIvRzgrZlp3U3NtMG1ZTkU4WUpVcXdDbWZEcTJxMDUydkNMcUkzVDYzbHNuVzhFTGIvU2FxWjlTOHZBcVBmdEZ1MVZhRHIrNXlBWmlDSlFBZE1CT3lOSHhxeU15S1pNQ0lDSklOWU1FTHpreGE2NXlNVldDSG0xT3V1RFB3ODlQRUdEYVh5Q1hzeit5Ly9ubEVGNEZBbmFNL0xpeHZNUVVsN2drYm1QNDZnT3BKQ2NSazNNZ3Mwc3VQcHB0bFRFdWNGWEFEZ0VUUGlNWTZ4U0R0V1RNelBSZFJNZjB1MWUreHY0U2FXSXZqWXVsckhYWHVUVzNncjRoRnNnWGN1RC9mZitPVml1ZnFaSnFNM2FRZ3RLLzc3amxKcnRyc0xOTmZFeFcydlJzakc1NEpIMGFvNW1WR2dnUk1pZW43Z2lxeVdWOVR4VXZ2azhNTElYRUt5Y2VlTE1NRTBObTZTcEcwSitrZkEveEVkdXh2RExoZGpQOHB4Y0dBck9RZTgrMHVXcTJMaTRSZG9VeThSTlpWSnV3aTJmWGIrVHFHMitwVk0yVHdDdjllTk13SEQ0ckM2YmdRN2FJT05SWFc5SUtDd0FIR3k3RmpZbHpDeU9RSEZ0aWJGK1lSK1ExRjdyY3dKa0piME80WExNMXhUaERxQktnMVVFV3ZiRUthRm9hZ0I0VnpmbllKcm1yY1RJbzNJSTdCdFNoc0JyREUyU0UwcExLQzNJdkFSejZrRHZ5aU1OZG5XSzN6YWpwbVdpTm9PSDVEWlBUdGJjL25oR2crSEhFMXZwZ2ovWWlwQlpGL01XWm14Zmk1ajNDWjNIOGV6eWZaellBZWw5RkRWeE1Na1k3dWZNaGxqbjRTREJSc1d4TDNFKzhDOGlUMStWRHdXNkZBSTNBbkgvemtYNEt5SG1CUFBncVFOckIxclhpbXJSZzJFNWlXMkVTV1FBRHduYkowNnRmc1RaN2FIVHErc0pXSlI5WUh5SXFvejBHZmZMOWJMTXlGa1pNMVNUQVNBb2MwMkxUV2I3RUhIVDE2MFFLQ04xTFprSFE1a0tIM29JSWhvSXRadDEwNGxFcmVaV3IyVDRKZkpuNXhxeHBUYXhRMjZhQ0JIWm5Xc2ViM2d5WDI0WThla2gyTVNZT3JjTE9rNkNlczIySU15Ly9mOFlzWWxXZmg0OWZWNFl2NytxbzFPcWtrUW8zcEx5RUJST2IxdUl5bjRPaXRQYWNwK3RLak93MXZNaWpmVE1yaUFMRnJqa1ozQUJMVVRJYnM5QTZDUGRxdkJJc1B0WFVyeVdva0drL2ZJbDFrNHdKaFMzSDRneWtibEdVMjNzSDVJbjJ1ZElpNjh2Mlk5QUZsSG5rYjFJR0lHSnFzamtFM21LSFpOdy9wc2tSenNMR1V5TWQ2ZGptMUNNWTRhYUhEK0NnZzBPTlNGZHZoaElKWG5VV1V6N3hyWFVRU3cxbE9CY0hHOWFpa0htNEZERTUyVUNqQTJ3RXdJM016bTNvcnRjaG13bmpwVDQzUEhFWm5CTW1ockNZNSt5K2JNV08ra0c2dkxyTmJiWkhoUEprL1lNSG1NdjJmbmcvdzVzZEU4TWtEc2QzeHZXekN3REl0dGkrV1VTaTVTVUNkOUFxMHdnYjBoOFJFbnpnbjQxMG8xZEpTcUFpYyt5MzQrMm5vcTVOZVZSRHhweDNvNURsTEpVYTBVekF6MnI4ZDFVa3RMKzhiVlJDekRENnR2SWxJa0NzcUsxQ2llR1Z0Q2lxS0dsUmJ5QXl3RFp6ZWlMMXU3MjU2bCtvSEJ3WVRJNnJvU0FOUFFEQ1FFVTNOSjM0Z09oQUIvSUtOYktkcW12MWFFdjhpQkswZDdQaFBZSXcrK2NodGJrTHQwM2hneS9ydnJnN0o4UFMvSStKcmRSaDhCaTVDeWlEOXk4V04ya0VsOVNYa2ttRlBiTlNabzA5SzQvMkZrcG55dGhnakNjOC9lT0o1VnhhQllwdmcrdlQzWGVjWXcxdWtuUjZnSDdJcU0xQzVXODkyZGQwOWNqM1NkSTI3eER6SFlOZnJVUEMzTGVDOFhQMllsLzZrdlVkOWV6NWxyUC95RDFxMG9GNE4zdzNqa3VXK1NraEJGcmFlZEJ2THBFTHVYVVh0T3pNRm9ha0RCaVYrbmZFdS9jYnRSay8wOW1seGVrZitPbDk5VUVHQWRBUXNudUh0WVNLS2NLUmh3NHdDVFV4eWdTMk5zQWpTV2trUWM1NzVuZnd5aXBLR2hyQUNuZ3J0Y3Z5aDlrTm85QUI5OXZQUGRrd1U5TnFXbENaZGwyRnJsRUFCczc2c1BjaVA4WFJ3R0F3cHFpOHd5SEN4TW9UbGp5U3p0OXBQTG80RTd1ZFZrZU9PYlRNQlcyUm5XVGNQVUJjT0hkeDQ4aHhHYjJpT1BtNXA5REFSYXdERVo3WnNlZzBiQ3dXYWtpZmlwL01rc2dsQ0xETko3aUptZjAvU3VBMVRCZGZmZDRkL201Uk1jdTk3Tmw4Q1ZCRksyRnA4TFVUMHM2aGZySlFIV2FpZVlmaVcyRStUbEJZZ3BsaDNSaG9EcXBXTlEvS2VIaWVrcGxEZ0xDZ0JjMnJQSlNtdWwwVmtSMHlhK1A4RThndzJqUEMrWTg4eDFiWmhDV1lZYTFOdDFXQTV4TFkyK0tGcHZvVUhhdFd4Nk9PWVlpODBSSW5JY0pCZW1oNUxzeDc3MVJwUnNvMlNPRWlrTGNwL3RycDZSZTJRc3l0eXdxUFhvNCsyRDNUOVNTc29sVXFSVUdmamFib2VGRkJUV1JINWlSdWV4bVFEbHNaK3VST3YrOFUvSXhHRXNYeHIvOVdNNFVKbGRqZGwwR0IzSVhJcGR4V0l6L25LWkw2OXVXUGR2NzhkTzRad25UOTI0M3dwRDZKbm55ZlVKWUxDeERGY0IrMmdybmFVMmpZV0tsRkpiVEVMS0tycGpIKzRRS3c2T2dpaGJMSkxKaE5WWU8xWGNxcFYvVUpTclNuU3ZTakdVcXhSSFZhaGZTOE4wQVdRRExRd2NRWHlGZ1hhN3BqWG41TmcxZE1oYy83STdQaGE1RkpYQnpoNStWYnpQaWtHMUN1bmZodCs2TXlIa3BIS2c3a2JxNzNyU0V3d2M3eHhjYjZLc0dOeHhMeHIvM1R2WEtoT28rZjZySTJtY2lrTERmZEZzMU1QUlg1VENjVXVSa1VuOXJQWHJ6OWhHUDcwK3FnVkRyZzhpQ0YvRVpxUWNwQytHL1d6OTU2T21UTTJPVDUwMXF1alRNeDNac2RZWlhsMktHRVAvdWhSQWtJeWxPSllDT1JVU2w5KzVVWDZNdy9DOU9vQ3A1OThENEhmL0dzdGd5MGllUVUyZXRacG41dFB2K3ZFVzBjUThoUWNMY0h3WGtkbXo0Wjg2UHloamFMWlQyMlFseTRnRTczcFU1VzF0VnFQUGNmeWV3SHoyby9mdG1yS3ZhQ0ZvZWkrcGRrNDBsc3FvVmFlbmRlRFNqamU5U2huVW9xR0lnU1BRTFB1d1RnV2hHWlhBUll4ME9nSHhOaTA4eWZXNDRSQ20vdVR3UUxoMjcwYm1hbFBldktPNjAzZkZwbS9wdWJXS0EwRWJhT1o4cG1GQ1k5QVg2Zkp6eU04YW5CcWpuSGRRUEhyb1Vad3AyWE1EUzQvc0tUdm1HNWo4MlFrTytxb0R3U2lMSU1STXUxVVBpanJ6NC9HeE5vcU4xc3NxRnc1bUVxZUpRdHZPZVB2TThKWFVINWErTEh0aDVMRW0rV1d3MXlraVJqTGJhQWNtNXhKZDhtOGNhczZpVlNaNkJGdk1xdDcxSWtJTzRrS0NYY2Z1RHJZS0xSdGxySHRJYkRuODlVYU9JeWhURlNXTkNiNVNFRnhTZm9XeGZzZnY2czZ2d3puT3NvZ29SYjJlQktaVXZiVHhPYnR0dXRheUpjMHRtTGVtOHVwTnRmNFJHUHhlNUp3emVWNTd2YW5tMVZaZlAvMFlQb01idXVUdEdqZDhyQW0wTUVPb3VJa25MUi9jbjdWVXZNVTZ6TzdJWHh5a2IzVlFLdmZiSUZ4ek9jZzV6UXR6V2JSVzlTVWo1akwrUWRuM0FPSmVXNmp5K3lBbnlyN00yYTJiR1N0RFI2NmxtTU9TT1o1WC91M2tBdjFtK0dscjA4WUpZZ1FwOVVZaVB5Qk9vaWtUbHUwalMwSCtQMDBzV2JWYTRieDNwZ1dDeUJhUUlueXdERVkxY21lUzZlVW55bWwzT3JiVFRiaHQ2cE1aaUtQTzhZNzM5U3BiVEF3bXBicVNVdDVvVm9MNU0rNHg0czFZL0dSMjdISmZKMjFYMy8xcExuRGp5SjYxbEZNMWZ3OXFQQXBRdXFHNEdmNTdoNzk1NFdMWlF6Ym4wZ0VlZ2N2Zi9NZkhvb1N5TCtMczN4UEQ3cE12alBsUGxwQlBFdXNGUDB3WDdtb1lRRUhrNFJMTURpZThhN1E4YXpwaHJGZDBsb3daK2FSQU44VkdtK2luakVwWjkwZjFzOC9VWkM4YURCNnN2eWY5V1M2TkYxN3hFaElwOGVPUy9JbmZRczd0TmlPenAyZjh2ZTQxbGk3RUxMaEVHY2RodkQzMnloYmlnY3RINnkxamNZUWlINXRpR0RocXloc0tPRGJuYXRYQlB0ZUNUTDgyMnpWeHFRV2JJc3YxSkVRU1JLNnRqSHlLZUZ4ZTFHcTBRNlkwUGhheXpMd0k5bFhJRG5UMzgrZE4vekpLTm9HRjd0U3NvSXBYVXQzNGttcEIwN0FHejZwb3J0UnJ1ZTZvdy9sem9Vc3JyRm1CZzZ1dmZNaEFWSlhRTGVMWWNjV3BnMisrNXBCTnBsdGRYQS9neW9vTmxYOVFaTnhQMGpKb2F2VFZ6TmIzNWgzaFZJQ1J1K051ZndEejlpSGUrYS92dmM2NjFhVzJTQnd6TDMvM3BmYy9BYVlyU3N5c2VGSEtYOUFvMkhRODVNRkdOZnNmNSsxdVl6SmF5UkZnMTloOWhybFlsbGVzMkxuSUswQVZkTm9vM3d4NmU0OXZ3RE5XODdYWHI4L3NlMHQ2MHJsalhyOHJnUnd2UGZOK2drcTIrZlFOQUtiNGs2eUdsc1JaRmYvWHU1ZXR2V3JrVTVTQjB4UmFHNFMwUytMakJ3Z0xQTlk1MitkQ2JEd3hZSWFkTVFGZ002WXFPMzFMUWhhcWl6QTBPOU1iTXpsUlhlWTM4ZDkwMVNXMTU3Znh6bTJPU2c1dU0zRnBZVHNJT3BCNC9VVUpOYnZzYVArN1BxVmxaSVczTGtUcGszeHY3b3FQbTZuU3pFeHdVa0EweXJ3N0lJWHBHelJPN0wyWjFFdFVobTgvbUdGQ0pTODlxZ2EvL25lMG1IRWs0Y3JyWW1wTzJlbisxMnNVRnRneGIxTnhnRm1rMlh0eXYrdDNJdXZCNWZGTWVtQWVNTC9qSWdDWnlXb1E3OGJ1SHhlT0VCa1lqNFBLZWNEclVMUy9HNG96bEhiclgvNmRZSC9JejR6UXR0TmJ6T0lqRXFvd0FBVWFFV1BMdXdpQXlCajI5R1JjTEdZV21zVDQvYWVqelBqVzlpOEpPc3RtYmxVVUkyMFk3dm5TOFVlZ1hkUUJtK3E0ZWFXcU50b3NGZ3c3SCtwWHljaER6OFgxK05rNjNmcDZFUUVEZU9ZVk1OTGcyc1JETnRuYTlzd2hYdDBWNkhPaDJjemNWclJ6ZStTMGFxT1RHbWs0RHFkaGtKanI3UythWm9lckw2YVlDVmFrazZ4UXN3MnBOTWRZS2FtYklJbmNQcWk0MUlqVmRCRnBZN1lVcjZBZXdDNzkxWlMzaUVvaWJ3OHV6c3UxZkFlRVBWVlF4QTA3Nkw5dXFGMEZGM21LYXVleWVTWWVPWXFyL1VDWGgybXp1RTBwb0laQitSWEhEVjB5Y1VqSGRDSzQ3UzZlVU5SMWpaZ3ZuZ0RKcFJ0dzBwVUVmNDJRekdUeVo5TTFoOUpJck1qNGJjMkZ2WWVsNm5qWDFOd2g0eXZVaEdyb0tHS3JXVFF4OUplek9ZWnlRdGtrYjFNVGlpMG1yRm5ERERtWk44VHNrZHBOT21IY3VtK3o4dWRmOU1sOGE5ZDZNVmxIUHRXNDc0OVFjMU02QlFuMDJJeGwvaVVNYlJudzRxc1I5alFTUG5CdkorYnZpN29STURCQVpSQVNKcUxacWF3dGtieFVQcjlpdG5ldTkyU1FGbFlGVW9YK1pQNlBFWmFTMWRzQlFBUkFNNmRGRGNtaFVRdVovWXBubWdpQWozUlV0Qk5rUWhDeXdXNXRXTktVeWVhVWpNdU5wMDdlZldGSGx0NDJPSW5mdHJNTk1ZK2UwaVo5TmlYYTg1VmlyZHdrT3dnc211dExuQnNBaDJ4SFRrcGYzQVpSdERSREdCOEVHWXd0a1ZZcEwvNDFsRGFjRTFjTU5VUm04b2c3WjV0RDkvUXhPblJ5RnZMdjVnVllhZmJqcHBhaExYa1p3WldVVDg1aG1aakRMOTdSam1OdWMwTjdwb010ZDNiQzhjWCtVZkF5aUd5K2NwbEhLN2RHVHJTaHNoWk54Mk1kSGc2ODN1ZTFZcHpOdWFJNjJaN2c4TGFZa1dlVnM2bWZxU21RVlQ4KzU0U1hiRjhxZ2Zoa2RWT3lkOWtmbjRhenpzdXloWTQ3Q05neGNPOU0zZ1FNdHJIZlpxMlZyM1lJc0tBMHJGNERMdDhHVlFsYkpORm5yaSthT1pxd1YzeWZGUS8vVVQ0cytQL2QwSWtxOE9mQXAvMG1vU0syY0p2YXlXM0pwWkZUQ3hMNytHZTFQQWV0NUFXZ0RJcHZ1YkNwLzNLRkI2ek1SSXhUckRnaXJaWnBKNmNRZ2VvV1o3NUw2MzBhNWVMMlhyaUswN0VGa2c0eWRtMzRQNmZDL3FtZGY5WlFLZzZQdFY3NmV3d2s3Yy9iV3ZMcXdTZGRucEU3U3lOK2RIeVpJQ3dwYXQxVTRMbWgwZmovZDhMYzV5WlhPYnI0dnVPQWNpci9QUGlSMG1FZkRxcXVZait0cktWOFZpK1F0R2ZQeklqS1NjbVJtbmtKdHpyUXBET0J5Nzh6dW9vdzF1Ni93RllZK2FtaVA1Z3FUeFhUL3Vvci9wMVhnWEV1ajVZek03eHA4MksyUHMwRzZOd1F3TUJFRHF4QjBRT2dxRStnOEI0VU1RQWJFelMyQndDZytvdmZhQTh3RFJJNUhXQ2pmSVJQcGVRbW1nYjRXVVRlQzlnb0tQeHU3eEgzb05icUJnVUZvM1lnWW8vOWtjQW9ZYlNWaEkyQ3dqc29Ba05zRzdobEVma0hZS09nOE4rOFIvd092UHhGL2hwUUZpUnVmb0JOZ0JJWHhDWkJLUXN4cEFLbzl4NC9DMHpPSXU0M3dDWkFpWCtBSkRBRGlyc0Ftd0FsWG9Ra0lRQzBzYU9Wd0xoUUN0SGlvQ09MdUE2eENWQmlzRWtnRmZRR2lmdm92eUFKUXdDVU9NeGYwazhGTmdTSmUwZTBpZThCY3hkQU55Z3hkcmNvQUNTQk9KQ0tKQnRjNWRmb3g3WGRWd2dPR3pvcXg3MGNlUkxkNUlzQitjZFpmdEY3czV5V3RsdHpMdHhLYThGbTh5Uy9sdE9tK0x2cHZVQVZoK1ZhRzBUY3F5YTNOTEJXWERBV0oySGpIR1B2NDFKYUdxaUNyNVlqOTY3dlFWdEYzcFllOFlkWFM3OHJyWkNCaXRKSlNVTGxncXh3MUhCbm1qTlBvaUk0d1pYdTNuenJMbDBrZWhIUUNHaVRqM2xOK3BKMXZUUU90Q2JkRGJlc0JGL3dlWnVqQURYbml3QnJRMTdXdnF5bCtYNzVEQ1lwcndpd3dzckxTaTlySWM2V2pybmZNMnUrR2g4N2k3S1N6alIzaTBCclQxN1dtYXl0K2NYSm03bVVLTE9uVUxTTzVtVmR6YnF4dmhLNmkvL1hPNHA0U0UvLzRmUTN0SXN0OGkzRE81MTFDTWoybFZWY1FyTno5NG93MHBhOXpzckwvdG10Q0FBQkZMRHArNzJCNmFOOC95WmgyQ3VBZDk1OFB3SEFoeC9YVi9oN3pudHdDMjRFTUJBS0lNQS9NbDdjSCtWLzJXNWpxOE4zbkYwcXVUdXliMlFmTHZIQW9Cb0pSSFpMUlUrZU50TUtId2xUeEtSQkorZnljbW9vanQ5Q1l6R1NkKy90RS9VYjVVS1dYU0pldk1pcHVtT05xZ0EvQzVaN1lJanZrcFFlUS9xbWZPWmZ2UmRhT210OVR6VGdsZDQ0dEV2YVJnUHUvWEQzOFVSSEFYNTlNMzRBN05BVGpkT2FOS241WGRIVkowMVRpbXNoMitUZEsvc2d0STJxZDBuTVZGSzUvVW5kVXRrYjJYMlMzZ3ZCbmJXNEoxb3R3L08zYjhWb0lLRWZybzhDQXZvbWVBQnMweFB4MDBxYzFLS3VTT3FUaENuRnQ1QnA1TjBEK3lDNGpXcDJpVzRxdFkxSzc1YklXY2p2dVd3dnpPeXMxVDB4b3pjdThXdWRHdzBJKytIK0tNQ20zU0xMckJzbnBKM3F5cGJ6TEIzY2JCbWxWa2MzS0lCUUl0UTk2cnNrMWY3dHcvTG5iMndsZ3Q0ZGd1aTNtTWwyNXFnQ0hGNm1kTWdIVDUrVitCcElCcTZYVjRIOFlyLzJmQzRGcTZ4Q3ljTUl2dWh3TUtWNG5zVjJGeDkzNzhPOGRqbnRWRnE2Nm1hTHppQWppZGJucitta0Z2dVZPWDhkS3hGdmc0emxLT3VvYVdXcXFvRkg5U1pVb0xwWWtRV3JxRUpod2VZT0I1UEd2cUhJOFJIVk1tbnROS2xxZGMwV2swRkNFbGxsTTNiU1JQc2xPWC9tbFlpMFFmQmthanFxc2t5bE5lQlhMNmNDMmNVcUsxZ3RNcG0xcHcwSGk0dm4xZXZhY2RFWE9ac2ZqVlU5ZEU1bzUvTm56RnFhYlZuWTExRnpIdlNoRmdzd2hHWlVZemNhN1V1cFBjZlEwTmNjdVpVVE9sZzYveHk3R1d1eVgyeUNBd1haWkw4RjJKMU9paHhZdENqQnJ0QkVTTG55NzZMZ2FpeGtiWTczdHFtMkJuSDRJckxxRUtjczg0d3d3ek8vdTF5Z3MyZWJxZVIzeURSbXk4ejRtcUZ4eCszc3BSRjdaMVJnZ1NIN3VHcGJjTGdiMHlkZ1ZwNWdyUmxUeEFPWStVa2pVbEE2LzR4Y3Z3MjlPbDlXdDI1ZUZhUkxrR0hQSGRVaVkwckcxbHRkcyttTldsMW1YWmV0V2JoWWtJSkhjcnhra3hMOVliUUUvRk9RZ2NaR0JSNHp2aTNJTFhnSnIrSTZYSVBuT2g2U3pJUVdjTUk5K1ZhRElDOHpIRndlbTUwaFo3ZHI0aEF0QnFlMHJuUzlvYWJBSUQweUFLNzFvTllWUVhPbEs0cnVlRmVNWEVkWHlGSk5WNXhBaWhCNlU5MkZUb2tXSU9DVUF0UUl0NDUvUnhubTVTdFFWU2FXWGNhYmExUXBXOFhrdTJkbHFxcnFXSlZSbktPMGpmeFpjVGlqRURxWmF1U200aXBqUk1tVnIxcXhzZ3B4ZTFEUnYxMXFJaUg1SjZtc3VPMnVJYndtaDgrT3hGRnFxRHc3N3JaNjIzR3BVNnBNdlFxdFRURnJHVGZLSW1LYUpQZU9xSWk0eWtvVjFlWHNYS1Bhb3FiK3FOVEV3YTVSNVpibkt5Um0xYklVMlJkcFNiT1JvM1J0aXVYS002SnA1NnVyN1BPN3JoeWhDdlc2dThsVnNxdjNuZE9aVUhCS2xpMnliYldRTFRzNTdMM21JTmRwWjUzanlNbVJPblBCcm1pYnpMK0k5UE04K0JWSnlxQ0RwK0x3OVA0N2QzUDZXeUFXa2VyL1AwVGVmUGg2UzQwU1VtbkR0R3pIOWZ4Nm85bHFkN3E5L21BNEdrK21zL2xpdVZwdnRydjk0WGc2WDY2MysrUDVlbisrY1BRS1RWQ3N5RVFsZWhnWWxmTDNSb0N5dDUrVWMxc2MzMVNsUnExcWVJdEZubjdVK1VYMUdrM1NvTmRrbVVmN2dISHhWcHpaNGlXWW9rbXpxWUFZOGNqZnliK1lHZ3BIb3JGNElwbEtaN0s1ZktGWW9yZjhXRWF6MWU1MGUvM0JjRFNlVEdmenhYSzEzbXgzKzhQeGRMNWNiL2ZINjl1N2xhdll5QUZ0SUlDUEg0cDZmSDQ1OE9YN3gxZjkvZnR2U0NNVDR4MHpreFZUbTBEOUNKakUrdUFvckxGb3Bja3dqaG9WelU0N01KcVV5VGJZNkpERFRqQmtGQjJIWlVxbXBrbXIvNHVKcHV0Rzk2OVJld2lKMk9nMElMTm1wRXdqUVRiTERIUE5NVStMZEs5b2FsbnJwbS9HWm03V1ptL081bTdlMmpaL0M3WlFtd2R1bVUvbmp2dHUxNzZPcjBiUTFCVzVOYmxRYlV1ckNOVGxLMlVSNkNvR2k4TVRpQ1F5aFVxak01Z3NOb2ZMNHd1RUluR0pHam85QUFBQUFBREFFbHdmWHFYV2FIVjZnOUZrdGxodGRvZXJtN3VIcDVlM2o2OWYrZCtXQUFBQUFBQUFpYVlNakxDK3lNUExWZW1GZmxDTXRaUmczdEJWeE5kZUtGdXRseW5QVVRTWDV6NWZTVDcyWEVyV3FnSWExMm95N1ZaSjdhY2FUV05xVGJuSjlhWnI5SlRLSHFzTEswckxDbks3ZXBRTlkwUHU1Wk9EK2FmVmpXMnpoRnRDb0hBdElWaUVWUlBITFMzekpBLzN6SnFlcUNTTFhEV1IxOUxtT1JYbnBFYlhLR216M0s5ZkxhZFBUVFo3ODlNeU5XVXZlVEtNaWpUbDlMMDg3Y1BmZ0RDRDFoNVY1S3h1OTdKNy9tMDZWQnNBQUE9PSkgZm9ybWF0KCd3b2ZmMicpOwogIHVuaWNvZGUtcmFuZ2U6IFUrMDAwMC0wMEZGLCBVKzAxMzEsIFUrMDE1Mi0wMTUzLCBVKzAyQkItMDJCQywgVSswMkM2LCBVKzAyREEsIFUrMDJEQywgVSswMzA0LCBVKzAzMDgsIFUrMDMyOSwgVSsyMDAwLTIwNkYsIFUrMjBBQywgVSsyMTIyLCBVKzIxOTEsIFUrMjE5MywgVSsyMjEyLCBVKzIyMTUsIFUrRkVGRiwgVStGRkZEOwp9Cjwvc3R5bGU+Cjwvc3ZnPgo=", + link: "", }; export const IMAGES_LOGIN = { - main_image: - "", - email: - "", - crm: - "", - project: - "", - search: - "", - ticket: - "", + loginSVG: + "", + buttonSVG: + "PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIiA/Pgo8IURPQ1RZUEUgc3ZnIFBVQkxJQyAiLS8vVzNDLy9EVEQgU1ZHIDEuMS8vRU4iICJodHRwOi8vd3d3LnczLm9yZy9HcmFwaGljcy9TVkcvMS4xL0RURC9zdmcxMS5kdGQiPgo8c3ZnIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgeG1sbnM6eGxpbms9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsiIHZlcnNpb249IjEuMSIgd2lkdGg9IjEwMDAiIGhlaWdodD0iMTIwIiB2aWV3Qm94PSIwIDAgMTAwMCAxMjAiIHhtbDpzcGFjZT0icHJlc2VydmUiPgo8cmVjdCBzdHlsZT0ic3Ryb2tlOiBfX1NUUk9LRV9fOyBzdHJva2Utd2lkdGg6IDU7IHN0cm9rZS1kYXNoYXJyYXk6IG5vbmU7IHN0cm9rZS1saW5lY2FwOiBidXR0OyBzdHJva2UtZGFzaG9mZnNldDogMDsgc3Ryb2tlLWxpbmVqb2luOiBtaXRlcjsgc3Ryb2tlLW1pdGVybGltaXQ6IDQ7IGZpbGw6IF9fRklMTF9fOyBmaWxsLXJ1bGU6IG5vbnplcm87IG9wYWNpdHk6IDE7IiB2ZWN0b3ItZWZmZWN0PSJub24tc2NhbGluZy1zdHJva2UiIHg9IjEwIiB5PSIxMCIgcng9IjE1IiByeT0iMTUiIHdpZHRoPSI5ODAiIGhlaWdodD0iMTAwIi8+Cjx0ZXh0IHg9IjUwJSIgeT0iNTAlIiBkb21pbmFudC1iYXNlbGluZT0ibWlkZGxlIiB0ZXh0LWFuY2hvcj0ibWlkZGxlIiB4bWw6c3BhY2U9InByZXNlcnZlIiBmb250LWZhbWlseT0iR29vZ2xlIFNhbnMgVGV4dCxHb29nbGUgU2FucyxSb2JvdG8sQXJpYWwsc2Fucy1zZXJpZiIgZm9udC1zaXplPSI1MCIgZm9udC1zdHlsZT0ibm9ybWFsIiBmb250LWFuY2hvcj0ibWlkZGxlIiBzdHlsZT0ic3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDE7IHN0cm9rZS1kYXNoYXJyYXk6IG5vbmU7IHN0cm9rZS1saW5lY2FwOiBidXR0OyBzdHJva2UtZGFzaG9mZnNldDogMDsgc3Ryb2tlLWxpbmVqb2luOiBtaXRlcjsgc3Ryb2tlLW1pdGVybGltaXQ6IDQ7IGZpbGw6IF9fQ09MT1JfXzsgZmlsbC1ydWxlOiBub256ZXJvOyBvcGFjaXR5OiAxOyB3aGl0ZS1zcGFjZTogcHJlOyIgPl9fVEVYVF9fPC90ZXh0Pgo8L3N2Zz4=", }; diff --git a/gmail/src/views/index.ts b/gmail/src/views/index.ts index 8e7f0825b..5d1de426d 100644 --- a/gmail/src/views/index.ts +++ b/gmail/src/views/index.ts @@ -1,34 +1,15 @@ import { buildPartnerView } from "./partner"; -import { buildErrorView } from "./error"; -import { buildCompanyView } from "./company"; -import { buildLoginMainView } from "./login"; import { buildCardActionsView } from "./card_actions"; +import { buildSearchPartnerView } from "./search_partner"; import { State } from "../models/state"; -import { actionCall } from "./helpers"; import { _t } from "../services/translation"; export function buildView(state: State) { const card = CardService.newCardBuilder(); - - if (state.error.code) { - buildErrorView(state, card); - } - - buildPartnerView(state, card); - - buildCompanyView(state, card); - - buildCardActionsView(state, card); - - if (!State.isLogged) { - card.setFixedFooter( - CardService.newFixedFooter().setPrimaryButton( - CardService.newTextButton() - .setText(_t("Login")) - .setBackgroundColor("#00A09D") - .setOnClickAction(actionCall(state, buildLoginMainView.name)), - ), - ); + if (state.searchedPartners?.length) { + return buildSearchPartnerView(state, "", false, _t("In this conversation"), true, true); + } else { + buildPartnerView(state, card); } return card.build(); diff --git a/gmail/src/views/leads.ts b/gmail/src/views/leads.ts index dea362508..f7e5bdb21 100644 --- a/gmail/src/views/leads.ts +++ b/gmail/src/views/leads.ts @@ -1,119 +1,128 @@ import { buildView } from "../views/index"; -import { pushCard, updateCard, createKeyValueWidget, actionCall, notify, openUrl } from "./helpers"; -import { URLS } from "../const"; +import { updateCard, createKeyValueWidget, actionCall, notify, openUrl } from "./helpers"; import { getOdooServerUrl } from "src/services/app_properties"; +import { getOdooRecordURL } from "src/services/odoo_redirection"; import { UI_ICONS } from "./icons"; import { logEmail } from "../services/log_email"; import { _t } from "../services/translation"; import { Lead } from "../models/lead"; import { State } from "../models/state"; +import { buildSearchRecordView } from "../views/search_records"; function onLogEmailOnLead(state: State, parameters: any) { const leadId = parameters.leadId; - if (State.checkLoggingState(state.email.messageId, "leads", leadId)) { - state.error = logEmail(leadId, "crm.lead", state.email); - if (!state.error.code) { - State.setLoggingState(state.email.messageId, "leads", leadId); + if (State.checkLoggingState(state.email.messageId, "crm.lead", leadId)) { + const error = logEmail(leadId, "crm.lead", state.email); + if (error.code) { + return notify(error.message); } + + State.setLoggingState(state.email.messageId, "crm.lead", leadId); return updateCard(buildView(state)); } - return notify(_t("Email already logged on the lead")); + return notify(_t("Email already logged on the opportunity")); } function onEmailAlreradyLoggedOnLead(state: State) { - return notify(_t("Email already logged on the lead")); + return notify(_t("Email already logged on the opportunity")); } function onCreateLead(state: State) { - const leadId = Lead.createLead(state.partner.id, state.email.body, state.email.subject); - - if (!leadId) { - return notify(_t("Could not create the lead")); + const result = Lead.createLead(state.partner, state.email); + if (!result) { + return notify(_t("Could not create the opportunity")); } - const cids = state.odooCompaniesParameter; - const leadUrl = - PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + - `/web#id=${leadId}&action=crm_mail_plugin.crm_lead_action_form_edit&model=crm.lead&view_type=form${cids}`; + const [lead, partner] = result; + state.partner = partner; + state.partner.leads.push(lead); + state.partner.leadCount += 1; + return updateCard(buildView(state)); +} - return openUrl(leadUrl); +function onSearchClick(state: State) { + return buildSearchRecordView( + state, + "crm.lead", + _t("Opportunities"), + _t("Log the email on the opportunity"), + _t("Email already logged on the opportunity"), + "revenuesDescription", + "", + true, + state.partner.leads, + ); } export function buildLeadsView(state: State, card: Card) { const odooServerUrl = getOdooServerUrl(); const partner = state.partner; - const leads = partner.leads; - - if (!leads) { + if (!partner.leads) { // CRM module is not installed // otherwise leads should be at least an empty array return; } + const leads = [...partner.leads].splice(0, 5); + const loggingState = State.getLoggingState(state.email.messageId); - const leadsSection = CardService.newCardSection().setHeader( - "" + _t("Opportunities (%s)", leads.length) + "", - ); - const cids = state.odooCompaniesParameter; + const leadsSection = CardService.newCardSection(); - if (state.partner.id) { - leadsSection.addWidget( - CardService.newTextButton().setText(_t("Create")).setOnClickAction(actionCall(state, onCreateLead.name)), - ); + const searchButton = CardService.newImageButton() + .setAltText(_t("Search Opportunities")) + .setIconUrl(UI_ICONS.search) + .setOnClickAction(actionCall(state, onSearchClick.name)); - for (let lead of leads) { - let leadRevenuesDescription; - if (lead.recurringRevenue) { - leadRevenuesDescription = _t( - "%(expected_revenue)s + %(recurring_revenue)s %(recurring_plan)s at %(probability)s%", - { - expected_revenue: lead.expectedRevenue, - probability: lead.probability, - recurring_revenue: lead.recurringRevenue, - recurring_plan: lead.recurringPlan, - }, + const title = partner.leadCount + ? _t("Opportunities (%s)", partner.leadCount) + : _t("Opportunities"); + const widget = CardService.newDecoratedText().setText("" + title + ""); + widget.setButton(searchButton); + leadsSection.addWidget(widget); + + const createButton = CardService.newTextButton() + .setText(_t("New")) + .setOnClickAction(actionCall(state, onCreateLead.name)); + + leadsSection.addWidget(createButton); + + for (let lead of leads) { + let leadButton = null; + if (loggingState["crm.lead"].indexOf(lead.id) >= 0) { + leadButton = CardService.newImageButton() + .setAltText(_t("Email already logged on the opportunity")) + .setIconUrl(UI_ICONS.email_logged) + .setOnClickAction(actionCall(state, onEmailAlreradyLoggedOnLead.name)); + } else { + leadButton = CardService.newImageButton() + .setAltText(_t("Log the email on the opportunity")) + .setIconUrl(UI_ICONS.email_in_odoo) + .setOnClickAction( + actionCall(state, onLogEmailOnLead.name, { + leadId: lead.id, + }), ); - } else { - leadRevenuesDescription = _t("%(expected_revenue)s at %(probability)s%", { - expected_revenue: lead.expectedRevenue, - probability: lead.probability, - }); - } - - let leadButton = null; - if (loggingState["leads"].indexOf(lead.id) >= 0) { - leadButton = CardService.newImageButton() - .setAltText(_t("Email already logged on the lead")) - .setIconUrl(UI_ICONS.email_logged) - .setOnClickAction(actionCall(state, onEmailAlreradyLoggedOnLead.name)); - } else { - leadButton = CardService.newImageButton() - .setAltText(_t("Log the email on the lead")) - .setIconUrl(UI_ICONS.email_in_odoo) - .setOnClickAction( - actionCall(state, onLogEmailOnLead.name, { - leadId: lead.id, - }), - ); - } - - leadsSection.addWidget( - createKeyValueWidget( - null, - lead.name, - null, - leadRevenuesDescription, - leadButton, - odooServerUrl + `/web#id=${lead.id}&model=crm.lead&view_type=form${cids}`, - ), - ); } - } else if (state.canCreatePartner) { - leadsSection.addWidget(CardService.newTextParagraph().setText(_t("Save Contact to create new Opportunities."))); - } else { + + leadsSection.addWidget( + createKeyValueWidget( + null, + lead.name, + null, + lead.revenuesDescription, + leadButton, + getOdooRecordURL("crm.lead", lead.id), + ), + ); + } + + if (leads.length < partner.leadCount) { leadsSection.addWidget( - CardService.newTextParagraph().setText(_t("You can only create opportunities for existing customers.")), + CardService.newTextButton() + .setText(_t("Show all")) + .setTextButtonStyle(CardService.TextButtonStyle["BORDERLESS"]) + .setOnClickAction(actionCall(state, onSearchClick.name)), ); } diff --git a/gmail/src/views/login.ts b/gmail/src/views/login.ts index 574432c08..eb3df3bf9 100644 --- a/gmail/src/views/login.ts +++ b/gmail/src/views/login.ts @@ -1,29 +1,46 @@ -import { formatUrl, repeat } from "../utils/format"; -import { notify, createKeyValueWidget } from "./helpers"; +import { formatUrl } from "../utils/format"; +import { notify } from "./helpers"; import { State } from "../models/state"; import { IMAGES_LOGIN } from "./icons"; -import { isOdooDatabaseReachable } from "../services/odoo_auth"; +import { getSupportedAddinVersion } from "../services/odoo_auth"; import { _t, clearTranslationCache } from "../services/translation"; import { setOdooServerUrl } from "src/services/app_properties"; function onNextLogin(event) { - const validatedUrl = formatUrl(event.formInput.odooServerUrl); + let validatedUrl = formatUrl(event.formInput.odooServerUrl); if (!validatedUrl) { return notify("Invalid URL"); } - if (!/^https:\/\/([^\/?]*\.)?odoo\.com(\/|$)/.test(validatedUrl)) { - return notify("The URL must be a subdomain of odoo.com"); + if (validatedUrl.endsWith("/odoo")) { + validatedUrl = validatedUrl.slice(0, -5); + } else if (validatedUrl.endsWith("/odoo/web")) { + validatedUrl = validatedUrl.slice(0, -9); + } else if (validatedUrl.endsWith("/web")) { + validatedUrl = validatedUrl.slice(0, -4); } + // TODO: uncomment the check before merge + // if (!/^https:\/\/([^\/?]*\.)?odoo\.com(\/|$)/.test(validatedUrl)) { + // return buildLoginMainView( + // "The URL must be a subdomain of odoo.com, see the documentation", + // ); + // } + clearTranslationCache(); setOdooServerUrl(validatedUrl); - if (!isOdooDatabaseReachable(validatedUrl)) { + const version = getSupportedAddinVersion(validatedUrl); + + if (!version) { + return notify("Could not connect to your database."); + } + + if (version !== 2) { return notify( - "Could not connect to your database. Make sure the module is installed in Odoo (Settings > General Settings > Integrations > Mail Plugins)", + "This addin version required Odoo 19.1 or a newer version, please install an older addin version.", ); } @@ -37,66 +54,80 @@ function onNextLogin(event) { .build(); } -export function buildLoginMainView() { +export function buildLoginMainView(error: string = null) { const card = CardService.newCardBuilder(); - // Trick to make large centered button - const invisibleChar = "⠀"; - - const faqUrl = "https://www.odoo.com/documentation/master/applications/productivity/mail_plugins.html"; - - card.addSection( - CardService.newCardSection() - .addWidget( - CardService.newImage().setAltText("Connect to your Odoo database").setImageUrl(IMAGES_LOGIN.main_image), - ) - .addWidget( - CardService.newTextInput() - .setFieldName("odooServerUrl") - .setTitle("Database URL") - .setHint("e.g. company.odoo.com") - .setValue(PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") || ""), - ) - .addWidget( - CardService.newTextButton() - .setText(repeat(invisibleChar, 12) + "Login" + repeat(invisibleChar, 12)) - .setTextButtonStyle(CardService.TextButtonStyle.FILLED) - .setBackgroundColor("#00A09D") - .setOnClickAction(CardService.newAction().setFunctionName(onNextLogin.name)), - ) - .addWidget(CardService.newTextParagraph().setText(repeat(invisibleChar, 13) + "OR")) - .addWidget( - CardService.newTextButton() - .setText(repeat(invisibleChar, 11) + " Sign Up" + repeat(invisibleChar, 11)) - .setOpenLink( - CardService.newOpenLink().setUrl( - "https://www.odoo.com/trial?selected_app=mail_plugin:crm:helpdesk:project", - ), + const loginButton = Utilities.base64Encode( + Utilities.newBlob(Utilities.base64Decode(IMAGES_LOGIN.buttonSVG)) + .getDataAsString() + .replace("__TEXT__", "Login") + .replace("__STROKE__", "#875a7b") + .replace("__FILL__", "#875a7b") + .replace("__COLOR__", "white"), + ); + + const signupButton = Utilities.base64Encode( + Utilities.newBlob(Utilities.base64Decode(IMAGES_LOGIN.buttonSVG)) + .getDataAsString() + .replace("__TEXT__", "Sign Up") + .replace("__STROKE__", "#e7e9ed") + .replace("__FILL__", "#e7e9ed") + .replace("__COLOR__", "#1e1e1e"), + ); + + const faqButton = Utilities.base64Encode( + Utilities.newBlob(Utilities.base64Decode(IMAGES_LOGIN.buttonSVG)) + .getDataAsString() + .replace("__TEXT__", "FAQ") + .replace("__STROKE__", "white") + .replace("__FILL__", "white") + .replace("__COLOR__", "#2f9e44"), + ); + + const section = CardService.newCardSection() + .addWidget( + CardService.newImage() + .setAltText("Connect to your Odoo database") + .setImageUrl(IMAGES_LOGIN.loginSVG), + ) + .addWidget( + CardService.newTextInput() + .setFieldName("odooServerUrl") + .setTitle("Connect to...") + .setHint("e.g. company.odoo.com") + .setValue( + PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") || "", + ), + ) + .addWidget( + CardService.newImage() + .setImageUrl("data:image/svg+xml;base64," + loginButton) + .setOnClickAction(CardService.newAction().setFunctionName(onNextLogin.name)), + ) + .addWidget( + CardService.newImage() + .setImageUrl("data:image/svg+xml;base64," + signupButton) + .setOpenLink( + CardService.newOpenLink().setUrl( + "https://www.odoo.com/trial?selected_app=mail_plugin:crm:helpdesk:project", ), - ) - .addWidget( - createKeyValueWidget(null, "Create leads from emails sent to your email address.", IMAGES_LOGIN.email), - ) - .addWidget( - createKeyValueWidget( - null, - "Create tickets from emails sent to your email address.", - IMAGES_LOGIN.ticket, ), - ) - .addWidget(createKeyValueWidget(null, "Centralize Prospects' emails into CRM.", IMAGES_LOGIN.crm)) - .addWidget( - createKeyValueWidget( - null, - "Generate Tasks from emails sent to your email address in any Odoo project.", - IMAGES_LOGIN.project, + ) + .addWidget( + CardService.newImage() + .setImageUrl("data:image/svg+xml;base64," + faqButton) + .setOpenLink( + CardService.newOpenLink().setUrl( + "https://www.odoo.com/documentation/master/applications/productivity/mail_plugins.html", + ), ), - ) - .addWidget(createKeyValueWidget(null, "Search and store insights on your contacts.", IMAGES_LOGIN.search)) - .addWidget( - CardService.newTextParagraph().setText(repeat(invisibleChar, 13) + `FAQ`), - ), - ); + ); + + if (error) { + section.addWidget(CardService.newTextParagraph().setText(error)); + } + + card.addSection(section); return card.build(); } diff --git a/gmail/src/views/partner.ts b/gmail/src/views/partner.ts index 3b0b3da71..435ebf5d8 100644 --- a/gmail/src/views/partner.ts +++ b/gmail/src/views/partner.ts @@ -1,113 +1,62 @@ -import { buildView } from "./index"; import { buildLeadsView } from "./leads"; import { buildTasksView } from "./tasks"; import { buildTicketsView } from "./tickets"; import { buildPartnerActionView } from "./partner_actions"; -import { updateCard } from "./helpers"; -import { UI_ICONS } from "./icons"; -import { createKeyValueWidget, actionCall, notify } from "./helpers"; -import { URLS } from "../const"; +import { actionCall, createKeyValueWidget, notify, updateCard } from "./helpers"; import { getOdooServerUrl } from "src/services/app_properties"; import { State } from "../models/state"; -import { Partner } from "../models/partner"; -import { ErrorMessage } from "../models/error_message"; -import { logEmail } from "../services/log_email"; import { _t } from "../services/translation"; -import { buildLoginMainView } from "./login"; +import { buildCardActionsView } from "./card_actions"; +import { Partner } from "src/models/partner"; +import { buildView } from "./index"; -function onLogEmail(state: State) { - const partnerId = state.partner.id; +export function onReloadPartner(state: State) { + const values = Partner.getPartner(state.partner.name, state.partner.email, state.partner.id); - if (!partnerId) { - throw new Error(_t("This contact does not exist in the Odoo database.")); - } + [state.partner, state.canCreatePartner, state.canCreateProject] = values; - if (State.checkLoggingState(state.email.messageId, "partners", partnerId)) { - state.error = logEmail(partnerId, "res.partner", state.email); - if (!state.error.code) { - State.setLoggingState(state.email.messageId, "partners", partnerId); - } - return updateCard(buildView(state)); + if (values[3].code) { + return notify(values[3].message); } - return notify(_t("Email already logged on the contact")); -} - -function onSavePartner(state: State) { - const partnerValues = { - name: state.partner.name, - email: state.partner.email, - company: state.partner.company && state.partner.company.id, - }; - const partnerId = Partner.savePartner(partnerValues); - if (partnerId) { - state.partner.id = partnerId; - state.searchedPartners = null; - state.error = new ErrorMessage(); - return updateCard(buildView(state)); - } else { - return notify(_t("Can not save the contact")); - } -} - -export function onEmailAlreadyLogged(state: State) { - return notify(_t("Email already logged on the contact")); + return updateCard(buildView(state)); } export function buildPartnerView(state: State, card: Card) { - const partner = state.partner; - const odooServerUrl = getOdooServerUrl(); - const canContactOdooDatabase = state.error.canContactOdooDatabase && State.isLogged; + card.addCardAction( + CardService.newCardAction() + .setText(_t("Refresh")) + .setOnClickAction(actionCall(state, onReloadPartner.name)), + ); - const loggingState = State.getLoggingState(state.email.messageId); - const isEmailLogged = partner.id && loggingState["partners"].indexOf(partner.id) >= 0; + buildCardActionsView(card); - const partnerSection = CardService.newCardSection().setHeader("" + _t("Contact") + ""); + const partner = state.partner; + const odooServerUrl = getOdooServerUrl(); - let partnerButton = null; - if (canContactOdooDatabase && !partner.id) { - partnerButton = state.canCreatePartner - ? CardService.newImageButton() - .setAltText(_t("Save in Odoo")) - .setIconUrl(UI_ICONS.save_in_odoo) - .setOnClickAction(actionCall(state, onSavePartner.name)) - : null; - } else if (canContactOdooDatabase && !isEmailLogged) { - partnerButton = partner.isWriteable - ? CardService.newImageButton() - .setAltText(_t("Log email")) - .setIconUrl(UI_ICONS.email_in_odoo) - .setOnClickAction(actionCall(state, onLogEmail.name)) - : null; - } else if (canContactOdooDatabase && isEmailLogged) { - partnerButton = CardService.newImageButton() - .setAltText(_t("Email already logged on the contact")) - .setIconUrl(UI_ICONS.email_logged) - .setOnClickAction(actionCall(state, onEmailAlreadyLogged.name)); - } else if (!State.isLogged) { - // button "Log the email" but it redirects to the login page - partnerButton = CardService.newImageButton() - .setAltText(_t("Log email")) - .setIconUrl(UI_ICONS.email_in_odoo) - .setOnClickAction(actionCall(state, buildLoginMainView.name)); - } + const partnerSection = CardService.newCardSection().setHeader( + "" + _t("Contact Details") + "", + ); - const partnerContent = [partner.email, partner.phone] + let partnerContent = [ + partner.companyName && `🏢 ${partner.companyName}`, + partner.email && `✉️ ${partner.email}`, + partner.phone && `📞 ${partner.phone}`, + ] .filter((x) => x) - .map((x) => `${x}`); - const cids = state.odooCompaniesParameter; + .map((x) => `${x}`) + .join("
"); + if (!partner.id) { + partnerContent = _t("New Person"); + } const partnerCard = createKeyValueWidget( null, - partner.name + "
" + partnerContent.join("
"), - partner.image || (partner.isCompany ? UI_ICONS.no_company : UI_ICONS.person), + partner.name || partner.email || "", + partner.getImage(), + partnerContent.length ? partnerContent : null, + null, null, - partnerButton, - partner.id - ? odooServerUrl + `/web#id=${partner.id}&model=res.partner&view_type=form${cids}` - : canContactOdooDatabase - ? null - : actionCall(state, buildLoginMainView.name), false, partner.email, CardService.ImageCropType.CIRCLE, @@ -119,7 +68,7 @@ export function buildPartnerView(state: State, card: Card) { card.addSection(partnerSection); - if (canContactOdooDatabase) { + if (State.isLogged) { buildLeadsView(state, card); buildTicketsView(state, card); buildTasksView(state, card); diff --git a/gmail/src/views/partner_actions.ts b/gmail/src/views/partner_actions.ts index f284f4c1a..9b802c801 100644 --- a/gmail/src/views/partner_actions.ts +++ b/gmail/src/views/partner_actions.ts @@ -3,67 +3,100 @@ import { buildSearchPartnerView } from "./search_partner"; import { UI_ICONS } from "./icons"; import { State } from "../models/state"; import { Partner } from "../models/partner"; -import { actionCall } from "./helpers"; +import { actionCall, notify } from "./helpers"; import { updateCard } from "./helpers"; import { _t } from "../services/translation"; -import { buildLoginMainView } from "./login"; +import { logEmail } from "../services/log_email"; +import { getOdooRecordURL } from "src/services/odoo_redirection"; -function onSearchPartner(state: State) { - if (!state.searchedPartners) { - const [partners, error] = Partner.searchPartner(state.partner.email); - state.searchedPartners = partners; +function onLogEmail(state: State) { + const partnerId = state.partner.id; + + if (!partnerId) { + throw new Error(_t("This contact does not exist in the Odoo database.")); } - return buildSearchPartnerView(state, state.partner.email, true); + if (State.checkLoggingState(state.email.messageId, "res.partner", partnerId)) { + const error = logEmail(partnerId, "res.partner", state.email); + if (error.code) { + return notify(error.message); + } + State.setLoggingState(state.email.messageId, "res.partner", partnerId); + return updateCard(buildView(state)); + } + return notify(_t("Email already logged on the contact")); } -function onReloadPartner(state: State) { - [ - state.partner, - state.odooUserCompanies, - state.canCreatePartner, - state.canCreateProject, - state.error, - ] = Partner.getPartner(state.partner.email, state.partner.name, state.partner.id); +function onSavePartner(state: State) { + const partner = Partner.savePartner(state.partner); + if (partner) { + state.partner = partner; + state.partner.isWritable = true; + state.searchedPartners = null; + return updateCard(buildView(state)); + } + return notify(_t("Can not save the contact")); +} - return updateCard(buildView(state)); +export function onEmailAlreadyLoggedContact(state: State) { + return notify(_t("Email already logged on the contact")); } -export function buildPartnerActionView(state: State, partnerSection: CardSection) { - const isLogged = State.isLogged; - const canContactOdooDatabase = state.error.canContactOdooDatabase && isLogged; +function onSearchPartner(state: State) { + state.searchedPartners = []; + return buildSearchPartnerView(state, state.partner.email, true); +} - if (canContactOdooDatabase) { - const actionButtonSet = CardService.newButtonSet(); +export function buildPartnerActionView(state: State, partnerSection: CardSection) { + const actionButtonSet = CardService.newButtonSet(); - if (state.partner.id) { - actionButtonSet.addButton( - CardService.newImageButton() - .setAltText(_t("Refresh")) - .setIconUrl(UI_ICONS.reload) - .setOnClickAction(actionCall(state, onReloadPartner.name)), - ); - } + const loggingState = State.getLoggingState(state.email.messageId); + const isEmailLogged = + state.partner.id && loggingState["res.partner"].indexOf(state.partner.id) >= 0; + if (!state.partner.id && state.canCreatePartner) { + actionButtonSet.addButton( + CardService.newTextButton() + .setText(_t("Add to Odoo")) + .setBackgroundColor("#875a7b") + .setOnClickAction(actionCall(state, onSavePartner.name)), + ); + } + if (state.partner.id) { + actionButtonSet.addButton( + CardService.newTextButton() + .setText(_t("View in Odoo")) + .setBackgroundColor("#875a7b") + .setOpenLink( + CardService.newOpenLink().setUrl( + getOdooRecordURL("res.partner", state.partner.id), + ), + ), + ); + } + if (state.partner.id && !isEmailLogged && state.partner.isWritable) { actionButtonSet.addButton( CardService.newImageButton() - .setAltText(_t("Search contact")) - .setIconUrl(UI_ICONS.search) - .setOnClickAction(actionCall(state, onSearchPartner.name)), + .setAltText(_t("Log email")) + .setIconUrl(UI_ICONS.email_in_odoo) + .setOnClickAction(actionCall(state, onLogEmail.name)), ); - - partnerSection.addWidget(actionButtonSet); - } else if (!isLogged) { - // add button but it redirects to the login page - const actionButtonSet = CardService.newButtonSet(); - + } + if (state.partner.id && isEmailLogged) { actionButtonSet.addButton( CardService.newImageButton() - .setAltText(_t("Search contact")) - .setIconUrl(UI_ICONS.search) - .setOnClickAction(actionCall(state, buildLoginMainView.name)), + .setAltText(_t("Email already logged on the contact")) + .setIconUrl(UI_ICONS.email_logged) + .setOnClickAction(actionCall(state, onEmailAlreadyLoggedContact.name)), ); - - partnerSection.addWidget(actionButtonSet); } + + actionButtonSet.addButton( + CardService.newImageButton() + .setAltText(_t("Search contact")) + .setIconUrl(UI_ICONS.search) + .setOnClickAction(actionCall(state, onSearchPartner.name)), + ); + + partnerSection.addWidget(actionButtonSet); } diff --git a/gmail/src/views/search_partner.ts b/gmail/src/views/search_partner.ts index 38aa65855..d67202936 100644 --- a/gmail/src/views/search_partner.ts +++ b/gmail/src/views/search_partner.ts @@ -2,20 +2,25 @@ import { logEmail } from "../services/log_email"; import { _t } from "../services/translation"; import { Partner } from "../models/partner"; import { ErrorMessage } from "../models/error_message"; -import { createKeyValueWidget, actionCall, pushCard, updateCard, notify } from "./helpers"; +import { actionCall, pushCard, updateCard, notify } from "./helpers"; import { buildView } from "./index"; import { State } from "../models/state"; -import { SOCIAL_MEDIA_ICONS, UI_ICONS } from "./icons"; -import { onEmailAlreadyLogged } from "./partner"; +import { UI_ICONS } from "./icons"; +import { onEmailAlreadyLoggedContact } from "./partner_actions"; +import { buildCardActionsView } from "./card_actions"; function onSearchPartnerClick(state: State, parameters: any, inputs: any) { - const inputQuery = inputs.search_partner_query; - const query = (inputQuery && inputQuery.length && inputQuery[0]) || ""; - const [partners, error] = query && query.length ? Partner.searchPartner(query) : [[], new ErrorMessage()]; + const query = inputs.search_partner_query || ""; + const [partners, error] = + query && query.length ? Partner.searchPartner(query) : [[], new ErrorMessage()]; + if (error.code) { + return notify(error.message); + } state.searchedPartners = partners; - return updateCard(buildSearchPartnerView(state, query)); + const card = buildSearchPartnerView(state, query); + return parameters.fixCard ? pushCard(card) : updateCard(card); } function onLogEmailPartner(state: State, parameters: any) { const partnerId = parameters.partnerId; @@ -24,41 +29,52 @@ function onLogEmailPartner(state: State, parameters: any) { throw new Error(_t("This contact does not exist in the Odoo database.")); } - if (State.checkLoggingState(state.email.messageId, "partners", partnerId)) { - state.error = logEmail(partnerId, "res.partner", state.email); - if (!state.error.code) { - State.setLoggingState(state.email.messageId, "partners", partnerId); + if (State.checkLoggingState(state.email.messageId, "res.partner", partnerId)) { + const error = logEmail(partnerId, "res.partner", state.email); + if (error.code) { + return notify(error.message); } + State.setLoggingState(state.email.messageId, "res.partner", partnerId); return updateCard(buildSearchPartnerView(state, parameters.query)); } return notify(_t("Email already logged on the contact")); } function onOpenPartner(state: State, parameters: any) { - const partner = parameters.partner; - const [newPartner, odooUserCompanies, canCreatePartner, canCreateProject, error] = Partner.getPartner( - partner.email, + const partner = Partner.fromJson(parameters.partner); + const [newPartner, canCreatePartner, canCreateProject, error] = Partner.getPartner( partner.name, + partner.email, partner.id, ); + if (error.code) { + return notify(error.message); + } const newState = new State( newPartner, canCreatePartner, state.email, - odooUserCompanies, null, null, canCreateProject, - error, ); return pushCard(buildView(newState)); } -export function buildSearchPartnerView(state: State, query: string, initialSearch: boolean = false) { +export function buildSearchPartnerView( + state: State, + query: string, + initialSearch: boolean = false, + header: string = "", + noLogIcon: boolean = false, + fixCard: boolean = false, +) { const loggingState = State.getLoggingState(state.email.messageId); const card = CardService.newCardBuilder(); - let partners = (state.searchedPartners || []).filter((partner) => partner.id); + buildCardActionsView(card); + + let partners = state.searchedPartners || []; let searchValue = query; if (initialSearch && partners.length <= 1) { @@ -72,16 +88,19 @@ export function buildSearchPartnerView(state: State, query: string, initialSearc CardService.newTextInput() .setFieldName("search_partner_query") .setTitle(_t("Search contact")) - .setValue(searchValue) - .setOnChangeAction(actionCall(state, onSearchPartnerClick.name)), + .setValue(searchValue), ); searchSection.addWidget( CardService.newTextButton() .setText(_t("Search")) - .setOnClickAction(actionCall(state, onSearchPartnerClick.name)), + .setOnClickAction(actionCall(state, onSearchPartnerClick.name, { fixCard })), ); + if (header?.length) { + searchSection.addWidget(CardService.newTextParagraph().setText(`${header}`)); + } + for (let partner of partners) { const partnerCard = CardService.newDecoratedText() .setText(partner.name) @@ -89,13 +108,13 @@ export function buildSearchPartnerView(state: State, query: string, initialSearc .setOnClickAction(actionCall(state, onOpenPartner.name, { partner: partner })) .setStartIcon( CardService.newIconImage() - .setIconUrl(partner.image || (partner.isCompany ? UI_ICONS.no_company : UI_ICONS.person)) + .setIconUrl(partner.getImage()) .setImageCropType(CardService.ImageCropType.CIRCLE), ); - if (partner.isWriteable) { + if (partner.isWritable && !noLogIcon) { partnerCard.setButton( - loggingState["partners"].indexOf(partner.id) < 0 + loggingState["res.partner"].indexOf(partner.id) < 0 ? CardService.newImageButton() .setAltText(_t("Log email")) .setIconUrl(UI_ICONS.email_in_odoo) @@ -108,19 +127,27 @@ export function buildSearchPartnerView(state: State, query: string, initialSearc : CardService.newImageButton() .setAltText(_t("Email already logged on the contact")) .setIconUrl(UI_ICONS.email_logged) - .setOnClickAction(actionCall(state, onEmailAlreadyLogged.name)), + .setOnClickAction(actionCall(state, onEmailAlreadyLoggedContact.name)), ); } if (partner.email) { - partnerCard.setBottomLabel(partner.email); + partnerCard.setBottomLabel(partner.id ? partner.email : _t("New Person")); } searchSection.addWidget(partnerCard); } if ((!partners || !partners.length) && !initialSearch) { - searchSection.addWidget(CardService.newTextParagraph().setText(_t("No contact found."))); + const noRecord = Utilities.base64Encode( + Utilities.newBlob(Utilities.base64Decode(UI_ICONS.no_record)) + .getDataAsString() + .replace("No record found.", _t("No record found.")) + .replace("Try using different keywords.", _t("Try using different keywords.")), + ); + searchSection.addWidget( + CardService.newImage().setImageUrl("data:image/svg+xml;base64," + noRecord), + ); } card.addSection(searchSection); diff --git a/gmail/src/views/search_records.ts b/gmail/src/views/search_records.ts new file mode 100644 index 000000000..f95685bdf --- /dev/null +++ b/gmail/src/views/search_records.ts @@ -0,0 +1,169 @@ +import { logEmail } from "../services/log_email"; +import { _t } from "../services/translation"; +import { actionCall, updateCard, notify, openUrl } from "./helpers"; +import { State } from "../models/state"; +import { UI_ICONS } from "./icons"; +import { getOdooRecordURL } from "src/services/odoo_redirection"; +import { searchRecords } from "../services/search_records"; + +function onSearchRecordClick(state: State, parameters: any, inputs: any) { + const model = parameters.model; + const modelDescription = parameters.modelDescription; + const fieldInfo = parameters.fieldInfo; + const query = inputs.query || ""; + + const [records, totalCount, error] = searchRecords(model, query); + if (error.code) { + return notify(error.message); + } + return updateCard( + buildSearchRecordView( + state, + model, + modelDescription, + parameters.emailLogMessage, + parameters.emailAlreadyLoggedMessage, + fieldInfo, + query, + false, + records, + totalCount, + ), + ); +} + +function onLogEmailRecord(state: State, parameters: any) { + const model = parameters.model; + const modelDescription = parameters.modelDescription; + const fieldInfo = parameters.fieldInfo; + const recordId = parameters.recordId; + const records = parameters.records; + const totalCount = parameters.totalCount; + + if (State.checkLoggingState(state.email.messageId, model, recordId)) { + const error = logEmail(recordId, model, state.email); + if (error.code) { + return notify(error.message); + } + State.setLoggingState(state.email.messageId, model, recordId); + return updateCard( + buildSearchRecordView( + state, + model, + modelDescription, + parameters.emailLogMessage, + parameters.emailAlreadyLoggedMessage, + fieldInfo, + parameters.query, + false, + records, + totalCount, + ), + ); + } + return notify(_t("Email already logged")); +} + +function onOpenRecord(state: State, parameters: any) { + const model = parameters.model; + const recordId = parameters.recordId; + return openUrl(getOdooRecordURL(model, recordId)); +} + +function onEmailAlreadyLoggedOnRecord(parameters: any) { + return notify(parameters.emailAlreadyLoggedMessage); +} + +export function buildSearchRecordView( + state: State, + model: string, + modelDescription: string, + emailLogMessage: string, + emailAlreadyLoggedMessage: string, + fieldInfo: string = "", + query: string = "", + initialSearch: boolean = false, + records: any[] = [], + totalCount: number = 0, +) { + const loggingState = State.getLoggingState(state.email.messageId); + + const card = CardService.newCardBuilder(); + let searchValue = query; + + const baseArgs = { + model, + modelDescription, + fieldInfo, + records, + totalCount, + emailAlreadyLoggedMessage, + emailLogMessage, + }; + + const searchSection = CardService.newCardSection(); + + searchSection.addWidget( + CardService.newTextInput() + .setFieldName("query") + .setTitle(_t("Search %s", modelDescription)) + .setValue(searchValue) + .setOnChangeAction(actionCall(state, onSearchRecordClick.name, baseArgs)), + ); + + searchSection.addWidget( + CardService.newTextButton() + .setText(_t("Search")) + .setOnClickAction(actionCall(state, onSearchRecordClick.name, baseArgs)), + ); + + for (let record of records) { + const recordCard = CardService.newDecoratedText() + .setText(record.name) + .setWrapText(true) + .setOnClickAction(actionCall(state, onOpenRecord.name, { model, recordId: record.id })); + + if (fieldInfo?.length && record[fieldInfo]) { + recordCard.setBottomLabel(record[fieldInfo]); + } + + recordCard.setButton( + loggingState[model].indexOf(record.id) < 0 + ? CardService.newImageButton() + .setAltText(emailLogMessage) + .setIconUrl(UI_ICONS.email_in_odoo) + .setOnClickAction( + actionCall(state, onLogEmailRecord.name, { + ...baseArgs, + recordId: record.id, + query, + }), + ) + : CardService.newImageButton() + .setAltText(emailAlreadyLoggedMessage) + .setIconUrl(UI_ICONS.email_logged) + .setOnClickAction( + actionCall(null, onEmailAlreadyLoggedOnRecord.name, { + emailAlreadyLoggedMessage, + }), + ), + ); + + searchSection.addWidget(recordCard); + } + + if ((!records || !records.length) && !initialSearch) { + const noRecord = Utilities.base64Encode( + Utilities.newBlob(Utilities.base64Decode(UI_ICONS.no_record)) + .getDataAsString() + .replace("No record found.", _t("No record found.")) + .replace("Try using different keywords.", _t("Try using different keywords.")), + ); + searchSection.addWidget( + CardService.newImage().setImageUrl("data:image/svg+xml;base64," + noRecord), + ); + } + + card.addSection(searchSection); + return card.build(); +} diff --git a/gmail/src/views/tasks.ts b/gmail/src/views/tasks.ts index 4acb78a86..3716b27da 100644 --- a/gmail/src/views/tasks.ts +++ b/gmail/src/views/tasks.ts @@ -1,27 +1,42 @@ import { buildView } from "../views/index"; import { buildCreateTaskView } from "../views/create_task"; -import { updateCard } from "./helpers"; +import { pushCard, updateCard } from "./helpers"; import { UI_ICONS } from "./icons"; import { createKeyValueWidget, actionCall, notify } from "./helpers"; -import { URLS } from "../const"; import { getOdooServerUrl } from "src/services/app_properties"; import { State } from "../models/state"; import { logEmail } from "../services/log_email"; import { _t } from "../services/translation"; -import { truncate } from "../utils/format"; +import { getOdooRecordURL } from "src/services/odoo_redirection"; +import { buildSearchRecordView } from "../views/search_records"; function onCreateTask(state: State) { - return buildCreateTaskView(state); + return pushCard(buildCreateTaskView(state)); +} + +function onSearchClick(state: State) { + return buildSearchRecordView( + state, + "project.task", + _t("Tasks"), + _t("Log the email on the task"), + _t("Email already logged on the task"), + "projectName", + "", + true, + state.partner.tasks, + ); } function onLogEmailOnTask(state: State, parameters: any) { const taskId = parameters.taskId; - if (State.checkLoggingState(state.email.messageId, "tasks", taskId)) { - logEmail(taskId, "project.task", state.email); - if (!state.error.code) { - State.setLoggingState(state.email.messageId, "tasks", taskId); + if (State.checkLoggingState(state.email.messageId, "project.task", taskId)) { + const error = logEmail(taskId, "project.task", state.email); + if (error.code) { + return notify(error.message); } + State.setLoggingState(state.email.messageId, "project.task", taskId); return updateCard(buildView(state)); } return notify(_t("Email already logged on the task")); @@ -34,55 +49,66 @@ function onEmailAlreradyLoggedOnTask() { export function buildTasksView(state: State, card: Card) { const odooServerUrl = getOdooServerUrl(); const partner = state.partner; - const tasks = partner.tasks; - - if (!tasks) { + if (!partner.tasks) { return; } + const tasks = [...partner.tasks].splice(0, 5); + const loggingState = State.getLoggingState(state.email.messageId); - const tasksSection = CardService.newCardSection().setHeader("" + _t("Tasks (%s)", tasks.length) + ""); - const cids = state.odooCompaniesParameter; + const tasksSection = CardService.newCardSection(); - if (state.partner.id) { - tasksSection.addWidget( - CardService.newTextButton().setText(_t("Create")).setOnClickAction(actionCall(state, onCreateTask.name)), - ); + const searchButton = CardService.newImageButton() + .setAltText(_t("Search Tasks")) + .setIconUrl(UI_ICONS.search) + .setOnClickAction(actionCall(state, onSearchClick.name)); - for (let task of tasks) { - let taskButton = null; - if (loggingState["tasks"].indexOf(task.id) >= 0) { - taskButton = CardService.newImageButton() - .setAltText(_t("Email already logged on the task")) - .setIconUrl(UI_ICONS.email_logged) - .setOnClickAction(actionCall(state, onEmailAlreradyLoggedOnTask.name)); - } else { - taskButton = CardService.newImageButton() - .setAltText(_t("Log the email on the task")) - .setIconUrl(UI_ICONS.email_in_odoo) - .setOnClickAction( - actionCall(state, onLogEmailOnTask.name, { - taskId: task.id, - }), - ); - } + const title = partner.taskCount ? _t("Tasks (%s)", partner.taskCount) : _t("Tasks"); + const widget = CardService.newDecoratedText().setText("" + title + ""); + widget.setButton(searchButton); + tasksSection.addWidget(widget); - tasksSection.addWidget( - createKeyValueWidget( - task.projectName, - truncate(task.name, 35), - null, - null, - taskButton, - odooServerUrl + `/web#id=${task.id}&model=project.task&view_type=form${cids}`, - ), - ); + const createButton = CardService.newTextButton() + .setText(_t("New")) + .setOnClickAction(actionCall(state, onCreateTask.name)); + tasksSection.addWidget(createButton); + + for (let task of tasks) { + let taskButton = null; + if (loggingState["project.task"].indexOf(task.id) >= 0) { + taskButton = CardService.newImageButton() + .setAltText(_t("Email already logged on the task")) + .setIconUrl(UI_ICONS.email_logged) + .setOnClickAction(actionCall(state, onEmailAlreradyLoggedOnTask.name)); + } else { + taskButton = CardService.newImageButton() + .setAltText(_t("Log the email on the task")) + .setIconUrl(UI_ICONS.email_in_odoo) + .setOnClickAction( + actionCall(state, onLogEmailOnTask.name, { + taskId: task.id, + }), + ); } - } else if (state.canCreatePartner) { - tasksSection.addWidget(CardService.newTextParagraph().setText(_t("Save the contact to create new tasks."))); - } else { + + tasksSection.addWidget( + createKeyValueWidget( + null, + task.name, + null, + task.projectName, + taskButton, + getOdooRecordURL("project.task", task.id), + ), + ); + } + + if (tasks.length < partner.taskCount) { tasksSection.addWidget( - CardService.newTextParagraph().setText(_t("The Contact needs to exist to create Task.")), + CardService.newTextButton() + .setText(_t("Show all")) + .setTextButtonStyle(CardService.TextButtonStyle["BORDERLESS"]) + .setOnClickAction(actionCall(state, onSearchClick.name)), ); } diff --git a/gmail/src/views/tickets.ts b/gmail/src/views/tickets.ts index 225cff886..a5f25ff05 100644 --- a/gmail/src/views/tickets.ts +++ b/gmail/src/views/tickets.ts @@ -2,100 +2,127 @@ import { buildView } from "../views/index"; import { updateCard } from "./helpers"; import { UI_ICONS } from "./icons"; import { createKeyValueWidget, actionCall, notify, openUrl } from "./helpers"; -import { URLS } from "../const"; import { getOdooServerUrl } from "src/services/app_properties"; import { State } from "../models/state"; import { Ticket } from "../models/ticket"; import { logEmail } from "../services/log_email"; import { _t } from "../services/translation"; +import { getOdooRecordURL } from "src/services/odoo_redirection"; +import { buildSearchRecordView } from "../views/search_records"; function onCreateTicket(state: State) { - const ticketId = Ticket.createTicket(state.partner.id, state.email.body, state.email.subject); + const result = Ticket.createTicket(state.partner, state.email); - if (!ticketId) { + if (!result) { return notify(_t("Could not create the ticket")); } - const cids = state.odooCompaniesParameter; - - const ticketUrl = - PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + - `/web#id=${ticketId}&action=helpdesk_mail_plugin.helpdesk_ticket_action_form_edit&model=helpdesk.ticket&view_type=form${cids}`; + const [ticket, partner] = result; + state.partner = partner; + state.partner.tickets.push(ticket); + state.partner.ticketCount += 1; + return updateCard(buildView(state)); +} - return openUrl(ticketUrl); +function onSearchClick(state: State) { + return buildSearchRecordView( + state, + "helpdesk.ticket", + _t("Tickets"), + _t("Log the email on the ticket"), + _t("Email already logged on the ticket"), + "", + "", + true, + state.partner.tickets, + ); } function onLogEmailOnTicket(state: State, parameters: any) { const ticketId = parameters.ticketId; - if (State.checkLoggingState(state.email.messageId, "tickets", ticketId)) { - state.error = logEmail(ticketId, "helpdesk.ticket", state.email); - if (!state.error.code) { - State.setLoggingState(state.email.messageId, "tickets", ticketId); + if (State.checkLoggingState(state.email.messageId, "helpdesk.ticket", ticketId)) { + const error = logEmail(ticketId, "helpdesk.ticket", state.email); + if (error.code) { + return notify(error.message); } + + State.setLoggingState(state.email.messageId, "helpdesk.ticket", ticketId); return updateCard(buildView(state)); } return notify(_t("Email already logged on the ticket")); } -function onEmailAlreradyLoggedOnTicket() { +function onEmailAlreadyLoggedOnTicket() { return notify(_t("Email already logged on the ticket")); } export function buildTicketsView(state: State, card: Card) { const odooServerUrl = getOdooServerUrl(); const partner = state.partner; - const tickets = partner.tickets; - - if (!tickets) { + if (!partner.tickets) { + // Helpdesk not installed + // (otherwise we would have an empty array) return; } + const tickets = [...partner.tickets].splice(0, 5); + const loggingState = State.getLoggingState(state.email.messageId); - const ticketsSection = CardService.newCardSection().setHeader("" + _t("Tickets (%s)", tickets.length) + ""); + const ticketsSection = CardService.newCardSection(); + + const searchButton = CardService.newImageButton() + .setAltText(_t("Search Tickets")) + .setIconUrl(UI_ICONS.search) + .setOnClickAction(actionCall(state, onSearchClick.name)); + + const title = partner.ticketCount ? _t("Tickets (%s)", partner.ticketCount) : _t("Tickets"); + const widget = CardService.newDecoratedText().setText("" + title + ""); + widget.setButton(searchButton); + ticketsSection.addWidget(widget); + + const createButton = CardService.newTextButton() + .setText(_t("New")) + .setOnClickAction(actionCall(state, onCreateTicket.name)); + ticketsSection.addWidget(createButton); + + for (let ticket of tickets) { + let ticketButton = null; + if (loggingState["helpdesk.ticket"].indexOf(ticket.id) >= 0) { + ticketButton = CardService.newImageButton() + .setAltText(_t("Email already logged on the ticket")) + .setIconUrl(UI_ICONS.email_logged) + .setOnClickAction(actionCall(state, onEmailAlreadyLoggedOnTicket.name)); + } else { + ticketButton = CardService.newImageButton() + .setAltText(_t("Log the email on the ticket")) + .setIconUrl(UI_ICONS.email_in_odoo) + .setOnClickAction( + actionCall(state, onLogEmailOnTicket.name, { + ticketId: ticket.id, + }), + ); + } - if (state.partner.id) { ticketsSection.addWidget( - CardService.newTextButton().setText(_t("Create")).setOnClickAction(actionCall(state, onCreateTicket.name)), + createKeyValueWidget( + null, + ticket.name, + null, + ticket.stageName, + ticketButton, + getOdooRecordURL("helpdesk.ticket", ticket.id), + ), ); + } - const cids = state.odooCompaniesParameter; - - for (let ticket of tickets) { - let ticketButton = null; - if (loggingState["tickets"].indexOf(ticket.id) >= 0) { - ticketButton = CardService.newImageButton() - .setAltText(_t("Email already logged on the ticket")) - .setIconUrl(UI_ICONS.email_logged) - .setOnClickAction(actionCall(state, onEmailAlreradyLoggedOnTicket.name)); - } else { - ticketButton = CardService.newImageButton() - .setAltText(_t("Log the email on the ticket")) - .setIconUrl(UI_ICONS.email_in_odoo) - .setOnClickAction( - actionCall(state, "onLogEmailOnTicket", { - ticketId: ticket.id, - }), - ); - } - - ticketsSection.addWidget( - createKeyValueWidget( - null, - ticket.name, - null, - null, - ticketButton, - odooServerUrl + `/web#id=${ticket.id}&model=helpdesk.ticket&view_type=form${cids}`, - ), - ); - } - } else if (state.canCreatePartner) { - ticketsSection.addWidget(CardService.newTextParagraph().setText(_t("Save the contact to create new tickets."))); - } else { + if (tickets.length < partner.ticketCount) { ticketsSection.addWidget( - CardService.newTextParagraph().setText(_t("The Contact needs to exist to create Ticket.")), + CardService.newTextButton() + .setText(_t("Show all")) + .setTextButtonStyle(CardService.TextButtonStyle["BORDERLESS"]) + .setOnClickAction(actionCall(state, onSearchClick.name)), ); }