• Hi,

    Here I’m stuck again… I need to pass by a manual installation of GoogleMap in my single template file (I know it exists but we don’t want to use the plugin for our case). So I paste the followed script (with my key) :

    <script async defer src="https://maps.googleapis.com/maps/api/js?key=hidden_key&callback=initMap"></script>

    I pasted it directly in DOM. If I enqueue the google map API script with the wp_enqueue_script function, the script doesn’t seem to be recognized anymore…

    Now I need to get some Markers parameters in custom fields of some posts returned by a WP_query request after that. So I presume I should use the wp_localize_script function.
    But it need an handle script. So the script which initialize the map and the markers should be external. I’m wrong until that ?

    <script async defer src="https://maps.googleapis.com/maps/api/js?key=AIzaSyAhE_CVUWwIOzXs-zD-vFdtgWBFWRBhQVY&callback=initMap"></script>
    					<?php wp_enqueue_script( 'gmap_metierville_script', get_stylesheet_directory_uri() . '/assets/js/google_map_metier_ville.js', array('jquery'), SSO_VERSION, true); ?>
    

    The problem is, if I enqueued an external script, or put it directly in DOM, it will always be loaded before the script of google map API…

    So I got an error “uncaught exception: InvalidValueError: initMap is not a function”

    I tried some callback functions but nothing worked.

    I’m totaly lost on how to found a way to use PHP datas in a script which should be loaded after the google map API script… I lost my day trying some JS tricks but nothing did the job at all. Can you help me with that ?

    Thank you in advance,

    N.C.

Viewing 5 replies - 16 through 20 (of 20 total)
  • Moderator bcworkz

    (@bcworkz)

    It’s difficult if you do not understand some of the terminology. “Callback”, noun – a term used by coders for a function that is passed as an argument to another process. This argument will then be called (back) at the appropriate time during that other process’ execution. A common use of callbacks is in JavaScript when an event “listener” as added to a particular element on a webpage, such as a mouse click on a submit button. When that listener is added, a callback is passed as an argument. When that mouse click event happens (or “fires”), the callback function is called. Typically the callback function does something necessary related to the event that fired. That’s a quick example of a callback.

    The setup for enqueuing scripts is much more involved. To really understand it, you need to understand in general how action and filter hooks in WP work. It is not easy to grasp at first, but once you do, you will find it’s really rather simple. I don’t think I can adequately explain it, though. I tried to find a good reference, but was not satisfied with what I found. Maybe this article will help. While not satisfactory to me, it’s better than most.

    In the case of enqueuing scripts with wp_enqueue_script(), this function is required to be within a callback for the action “wp_enqueue_scripts”. Devs would normally add such callbacks (using add_action()) for a plugin or theme. By calling add_action() in plugin or theme code, WP stores the passed callback function in an array keyed by the action tag (“wp_enqueue_scripts” in our case). When themes and plugins are loaded for WP, WP is still in an unstable state. Once WP is at a place where it can properly handle calls to wp_enqueue_script(), if fires the “wp_enqueue_scripts” action by calling do_action('wp_enqueue_scripts');

    The do_action() function retrieves all functions stored in that array under the key “wp_enqueue_scripts” and calls each function listed there (sometimes referred to as an action “stack”) in turn by calling call_user_func() for each saved function. These callbacks for “wp_enqueue_scripts” could do any number of things, but generally call wp_enqueue_script() one or more times.

    That covers your part of the process. What WP does with the information is rather convoluted. Understanding what happens is helpful for deeper understanding, but many choose to simply take it on faith that it works, that the enqueued scripts are output to the web page in the necessary order. For anyone interested, here is what happens.

    What wp_enqueue_script() does is it saves the passed information in yet another array. After all callbacks for “wp_enqueue_scripts” are called and all scripts are saved into that array, WP processes the information, creating a list of HTML script tags in a particular order based on the listed dependencies. Scripts that many others are dependent on, but themselves have no dependencies will be listed first. These are typically libraries like jQuery or the maps API. Scripts that have multiple or no dependencies are generally listed last. Several registered libraries that are included with the WP distro, mostly related to jQuery, are all handled by a single script tag that references /wp-admin/load-scripts.php, followed by a list of registered handles that need to be loaded for the current web page. All other scripts will have their own script tag for each file.

    When a web page is output, there should always be a call to wp_head() on one of the page’s template files. This generally occurs in the header.php file of every theme. The call must be made before the losing HTML head tag of the page. All the wp_head() function does is call do_action('wp_head'); By firing the ‘wp_head’ action, all sorts of preliminary callbacks are executed that set things up for page output. One of these callbacks, added by WP itself internally, is wp_print_scripts(). This function takes the list of script tags previously established and outputs them. This causes the browser to retrieve the related files as part of the page loading process.

    Whew! Aren’t you glad you don’t have to manage all of that? To quickly review, here is an outline of the enqueuing process:

    Theme or plugin adds callback to “wp_enqueue_scripts”

    That callback in part calls wp_enqueue_script(), your part is done.
    ————
    After WP loads, it fires the “wp_enqueue_scripts” action

    Callbacks that were added execute and call wp_enqueue_script()

    wp_enqueue_script() adds script data to an array.

    After the “wp_enqueue_scripts” action completes, WP works out the dependencies

    When the page loads, wp_head() is called, firing “wp_head” action

    One callback is wp_print_scripts(), which outputs script tags in the correct order

    I hope this helps. I’m sorry it’s so long. I fear it may only serve to confuse things. All you really need to know is that you enqueue scripts from a callback to “wp_enqueue_scripts” (admin and login areas have their own hooks). The rest provides a deeper understanding, but fully understanding it is not required. Just trust that it works.

    Thread Starter Nokaa

    (@nokaa)

    Thank you for thoses clean explanations, I better understand how the enqueue function is working now.

    However, I didn’t found yet how to resolve the main problem…
    In this example : https://codex.www.ads-software.com/Function_Reference/wp_localize_script#example all the functions are called in the same file. So it’s an easy way to present the issue. But here members let me know that it’s better to enqueue or register our script in the functions.php file.

    I tried to enqueue my script in the functions.php (footer false), and use the localize function in a custom-post-type file where I take my array of data from a wp_query. But it doesn’t work, and I presume that’s because the script is not enqueued in the same file where I use the wp_localize_script function…

    If I register my scripts in the same file like the example, it works, but apparently I am not supposed to proceed so… There’s where I’m really stuck.

    Moderator bcworkz

    (@bcworkz)

    Like I tried to explain in my post before last, localizing in a template file is too late, the “wp_enqueue_scripts” action has already fired. As the localize function is just like the enqueue function, it needs to be called within a callback hooked to “wp_enqueue_scripts”. You should be able to see now after my last post why this is so.

    What data do you need to localize that is needed from the query? You would need to get it much earlier than on the template. Is this the main query for the page or a custom query on the template? If the latter, you would either need to create the query much earlier or not localize, rather output the JS data inside script tags. This ought to be OK since your map script doesn’t run until after the page’s DOM has loaded. I’ve had no need to get data this way, so I’m not sure how well that would work.

    If it’s the main query, there is likely an action within the query where the data is available early enough that “wp_enqueue_scripts” hasn’t yet fired where you can still localize. Even though in a different callback, localizing early enough ought to work fine. Hopefully you can now see how this is so.

    Thread Starter Nokaa

    (@nokaa)

    Oh my bad, seems that my answer was not well sended yesterday.

    What data do you need to localize that is needed from the query? […] Is this the main query for the page or a custom query on the template?

    In fact, in the middle of my template, I call a wp_query which return a list of profils (a custom post-type) which are associated with the region or the postal codes of the current custom post-type. So the result of the wp_query totally depend on the current post, so it need to be called in the template.

    Thoses profils have custom fields that I need to create the markers on my map (like the address and the city which enable me to get the location with the google geocoder, the name, the forname, the mark, etc.). I’ve my own script to do that, so it need to be loaded after the googlemap API script. And I need to localize thoses data in my own script.

    Moderator bcworkz

    (@bcworkz)

    No worries. It’s almost impossible to tell if a post not appearing is due to the forum’s system thinking it’s spam or it’s some other communications glitch where your post never gets to the forum. Most of the time it’s the former and your post will be cleared from the spam queue within an hour or two. Reposting before that happens results in duplicate posts and you end up looking rather silly through no fault of your own. But the latter means no one sees anything until you discover nothing is there and finally repost. So thanks for checking back and reposting ??

    Localizing after a custom query made on the template is much too late. All enqueued scripts and earlier localized data has already been output. Since this data is missing, your map script is likely erroring out due attempting to use undefined variables, so nothing happens at all. Your main map script ought to initiate an empty map so there is something to add the data to.

    During the template query loop, assign the needed map data directly by outputting JS variable assignments inside of script tags. Once the data has been assigned, a brief script to add the data to the map ought to be OK without enqueuing or localizing anything.

    The only alternative is to run the query very early, capturing output in a buffer. The map data can then be localized early enough. When appropriate, the buffered output can be sent to the browser with a single echo statement.

Viewing 5 replies - 16 through 20 (of 20 total)
  • The topic ‘Can’t find a way to load my script after Google Maps API’ is closed to new replies.