-
Notifications
You must be signed in to change notification settings - Fork 116
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[ADD] html_builder: convert s_google_map #4242
base: master-mysterious-egg
Are you sure you want to change the base?
Conversation
This PR targets the un-managed branch odoo-dev/odoo:master-mysterious-egg, it needs to be retargeted before it can be merged. |
c69d4b5
to
fe40bfa
Compare
f468b7f
to
80b3d5d
Compare
27a1402
to
8c6b756
Compare
50652bf
to
fc4217f
Compare
fc4217f
to
1073222
Compare
Steps to reproduce: - Activate developer mode - Install website Google Maps module - Navigate to Website in edit mode - Drop the google maps snippet - Enter a correct API key - The dialog closes but the snippet is removed This is due to a promise not resolved when confirming the API key.
ad1ffe4
to
ed7059b
Compare
This PR targets the un-managed branch odoo-dev/odoo:master-mysterious-egg, it needs to be retargeted before it can be merged. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ready for review!
I left a few comments in review for whoever will review this. This was not a straight forward conversion. I tried to preserve the original implementation in general where possible. Nothing should be missing but it's hard to guarantee given the number of pieces interacting for this in the original code.
@@ -273,8 +273,7 @@ export class WebsiteSnippetsMenu extends weSnippetEditor.SnippetsMenu { | |||
onMounted(() => this.props.onMounted(this.modalRef)); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This commit comes from this PR.
// @TODO mysterious-egg: the `google_map service` is a duplicate of the | ||
// `website_map_service`, but without the dependency on public | ||
// interactions. These are used only to restart the interactions once | ||
// the API is loaded. We do this in the plugin instead. Once | ||
// `html_builder` replaces `website`, we should be able to remove | ||
// `website_map_service` since only google_map service will be used. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@todo mysterious-egg
// @TODO mysterious-egg: should we still do this, which was just | ||
// done as a result of reporting a critical error? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@todo mysterious-egg
cursor: pointer; | ||
} | ||
|
||
/* Remove Google Map's own icon. */ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The whole file is just a duplicate from the original snippet, except for this rule. In the current website builder, Google Maps' Autocomplete doesn't inject its style into the document (probably because it does so in the iframe instead) so we style from a blank slate. Here however we need to revert the icon styles so we don't end up with two icons.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It s maybe because we don t use the custom css. We try to use bootstrap if it s possible. The website builder sidebar style is not working correctly. A designer will probably try to make it better and use bootstrap
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Are you saying I shouldn't bring in this css and I should try to do all this with bootstrap instead?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
No wait, what am I saying, that's not possible, this is not in the sidebar, it's Google stuff in position absolute in a div somewhere in the body.
@@ -27,7 +27,7 @@ import { GoogleMapOption } from "./google_map_option"; | |||
* @typedef {{ isValid: boolean, message?: string }} ApiKeyValidation | |||
*/ | |||
|
|||
class GoogleMapOptionPlugin extends Plugin { | |||
export class GoogleMapOptionPlugin extends Plugin { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Here are a number of small changes brought to make it easier to test.
@@ -7,11 +7,11 @@ import { Component, useState, useRef } from "@odoo/owl"; | |||
* @typedef {import('./google_map_option_plugin.js').ApiKeyValidation} ApiKeyValidation |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I made this commit separately to be able to revert it, in case. But it drives me crazy that in most places Google Maps was names Google Map :-P However, I kept the class name the same ("s_google_map") to avoid confusion.
@@ -0,0 +1,51 @@ | |||
import { _t } from "@web/core/l10n/translation"; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
i'm not really reviewing this, but i have a quick remark: all this code should go into html_builder/static/src/website_builder
. because it will be moved later in the website addon.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah ok! My whole google_maps_option
folder?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
yes, all that code should stay together
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
the html builder addon will become the generic foundation for website builder, mass mailing, report editor. so the goal is to have common code here, and specialized code in each corresponding addon
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Understood, done. What about the tests though?
@@ -273,8 +273,7 @@ export class WebsiteSnippetsMenu extends weSnippetEditor.SnippetsMenu { | |||
onMounted(() => this.props.onMounted(this.modalRef)); | |||
} | |||
onClickSave() { | |||
this.props.confirm(this.modalRef, this.state.apiKey); | |||
this.props.close(); | |||
this.props.confirm(this.modalRef, this.state.apiKey, this.props.close); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
why not
await this.props.confirm(...);
this.props.close()
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is just a cherry-pick of odoo#204139 so we don't have to wait for it to be merged to be able to use the snippet in the legacy website builder. It was mostly useful for me to test but I guess I could now remove the commit actually.
* @typedef {import('./google_map_option_plugin.js').ApiKeyValidation} ApiKeyValidation | ||
*/ | ||
|
||
export const GoogleMapApiKeyDialog = class extends Component { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
export class GoogleMapApiKeyDialog extends Component {
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done.
this.googleMapsAutocomplete = new google.maps.places.Autocomplete( | ||
inputEl, | ||
{ types: [ 'geocode' ] }, | ||
); | ||
google.maps.event.addListener( | ||
this.googleMapsAutocomplete, | ||
'place_changed', | ||
this.onPlaceChanged.bind(this), | ||
); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I m not sure it will work with the history. I don t know what is done by google.maps.places.Autocomplete
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It's initializing the API, attaching it to the input element. This is in the sidebar option though, does the history have something to do with it?
cursor: pointer; | ||
} | ||
|
||
/* Remove Google Map's own icon. */ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It s maybe because we don t use the custom css. We try to use bootstrap if it s possible. The website builder sidebar style is not working correctly. A designer will probably try to make it better and use bootstrap
}, | ||
}, | ||
showDescription: { | ||
apply: ({ editingElement }) => { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
you need to define isApplied:() => editingElement.querySelector(".description")
clean: () => descriptionEl.remove()
apply () => append(...)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
showDescription: {
isApplied: ({ editingElement }) => !!editingElement.querySelector(".description"),
apply: ({ editingElement }) => {
editingElement.append(renderToElement("html_builder.GoogleMapDescription"));
},
clean: ({ editingElement }) => {
editingElement.querySelector(".description").remove();
},
},
Nice. Done.
import { markup } from "@odoo/owl"; | ||
import { escape } from "@web/core/utils/strings"; | ||
|
||
registry.category("services").add("google_map", { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
it s just a copy paste or you need to change something ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Mostly a copy paste. Only change is I removed the dependency to "public.interactions":
odoo/addons/website/static/src/core/website_map_service.js
Lines 10 to 12 in ed7059b
dependencies: ["public.interactions", "notification"], | |
start(env, deps) { | |
const publicInteractions = deps["public.interactions"]; |
odoo/addons/website/static/src/core/website_map_service.js
Lines 43 to 46 in ed7059b
for (const el of document.querySelectorAll("section.s_google_map")) { | |
publicInteractions.stopInteractions(el); | |
publicInteractions.startInteractions(el); | |
} |
this.mapsAPI = google.maps; | ||
this.placesAPI = google.maps.places; | ||
const place = await this.nearbySearch(editingElement, editingElement.dataset.mapGps); | ||
if (place?.formatted_address) { | ||
editingElement.dataset.pinAddress = place.formatted_address; | ||
} | ||
if (!place && !this.isGoogleMapsErrorBeingHandled) { | ||
// Somehow the search failed but Google didn't trigger an error. | ||
this.dependencies.remove.removeElement(editingElement); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It look strange to edit the website dom during the willStart of a Component. You will have issue with the history
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You maybe want to add it when we add the google map snippet ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You maybe want to add it when we add the google map snippet ?
That triggers a whole domino of weird issues*.
This is just removing the whole snippet in case of critical error though (it used to be this.trigger_up('user_value_widget_critical')
). Is that so problematic?
* First one being that initializeAutocomplete
is called before the google
object is initialized. I can get around that by removing the on_add_element_handlers
from BuilderOptionsPlugin
as we had discussed so the option is not immediately started when dropping the snippet. But then if I await
loadGoogleMaps
in on_add_element_handlers
it waits forever instead of opening the API key dialog so I guess something would have to be done in the snippet dropping plugin to prevent that. If I don't await
it somehow the snippet has a <br>
in it instead of the map - no idea why. My point is this little change triggers a whole mess of issues to deal with that seem to be to be quite a bit larger than the snippet, and more foundational.
ed7059b
to
1c70092
Compare
addons/html_builder/static/src/plugins/google_map_option/google_map_option_plugin.js
Outdated
Show resolved
Hide resolved
* it sends a signal that the place changed. | ||
*/ | ||
onPlaceChanged() { | ||
this.props.commitPlace(this.googleMapsAutocomplete.getPlace(), this.state.map); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Detail:
why is the prop have an imperative name? It means "GoogleMapOption" knows the parent implementation.
I would rather call the method this.props.onPlaceChanged
to help with the mental model.
But it's up to you, feel free to resolve my comment and keep your code if you prefer.
* @param {Element} editingElement | ||
* @returns {Promise<void>} | ||
*/ | ||
async onGoogleMapsLoaded(editingElement) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I suggest to change the method name that tell what the method is doing? updateElementMap
?
Actually, I'm not sure what is the method supposed to do as I don't understand why we set pinAdress or remove the element. Reading the doc of the method did not help me understand what it is supposed to do or in which flow it should be called.
Why is it loadGoogleMaps
calling this method?
It look like we want to update an element that need to load google map if I understand correctly.
async updateElementMap(editingElement, forceReconfigure) {
const loaded = await this.loadGoogleMaps(editingElement, forceReconfigure)
if (!loaded) return;
...
}
} | ||
const place = await this.nearbySearch(editingElement, editingElement.dataset.mapGps); | ||
if (place?.formatted_address) { | ||
editingElement.dataset.pinAddress = place.formatted_address; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The mutation could be problematic with the history (same with mother other mutations in this PR)
Do we want a new step to be created?
If so, you should use the mutex (OperationPlugin.next)
If we don't want mutation and the attribute to be saved, add it in system_attributes
.
Do we want it in the dom without a step? In that we need to think more about that case.
Conversion of the Google Maps ("s_google_map") snippet.