【问题标题】:How to make PHP function run after javascript ends?javascript结束后如何使PHP函数运行?
【发布时间】:2011-05-06 15:45:31
【问题描述】:

我有两个 PHP 函数 - 一个构建 html 表单,另一个在提交时处理它。

我想在 html 中添加一个 javascript onClick 函数,该函数将表单的一个字段发送到外部 API(Google 地图)并将回复保存在表单的隐藏字段中,以便处理 PHP 函数将获取该数据也是。

我的问题是 - 如何确保处理 PHP 函数仅在 onClick 函数完成后触发?

或者我不能,我必须使用 ajax?

【问题讨论】:

  • @Pekka:类似,但几乎不是“完全重复”。
  • @Tomalak 我的理由是,这个问题几乎每天都会以数百种不同的形式出现,而答案总是同样无聊“不可能,使用 AJAX”。我们需要某种参考问题来关闭这些作为副本。如果我们找不到一个“纯”的,我们可能需要启动一个
  • @Pekka:“还有其他问题的答案相同”与“这完全与另一个问题相同”不同。
  • 我认为这里的问题有些混乱。措辞 OP 的问题确实听起来他们正在模糊服务器和客户端代码之间的界限,但他们提出的使用 JavaScript 查询 API 的策略(当然,AJAX 会在这里发挥作用)并将结果存储在一个隐藏的表单域在正确的轨道上。听起来真正的问题是:“如何确保我的 JavaScript 在表单提交之前完成运行?”

标签: php javascript google-maps


【解决方案1】:

您需要 2 个事件来完成此操作。

  1. 您的按钮的 onClick 事件,用于执行 google 地图请求并将数据保存到本地表单中
  2. onSubmit 表单的事件。您将使用此事件来查看表单是否可提交。基本上,在允许表单提交之前检查以确保您的谷歌地图请求已运行并已完成。

例子:

<script>
var googleMapsDone = false;
$('#gmap_button').click(function(event)
{
    doGoogleMapThing(function()//callback from googlemaps
    {
        //save data to local form for processing by script once posted
        googleMapsDone = true;
    });
});

$('#form').submit(function(event)
{
    //check to see if we did our google maps stuff, and that its done
    if (false == googleMapsDone)
    {
        event.preventDefault();
        return false;
    }
});
</script>

使用该代码,任何时候用户等待谷歌地图并点击提交,都不会发生任何事情。他们将不得不等待 GMaps 的响应,然后单击提交。这对于某些事情是可以的,但是如果您尝试对不需要用户输入/交互/反馈的 GMap 进行后台请求(可能在提交表单时获取其地址的 Long/Lat),那么您可以修改代码当您收到回复时,可以发布一点。一个例子是:

<script>
var googleMapsDone = false, submitWaiting = false;
$('#gmap_button').click(function(event)
{
    doGoogleMapThing(function()//callback from googlemaps
    {
        //save data to local form for processing by script once posted
        googleMapsDone = true;

        /* check to see if submitWaiting is TRUE.  If it is, automatically
           post the form when we get the response.
        */
        if (submitWaiting)
        {
            $('#form').submit();
        }
    });
});

$('#form').submit(function(event)
{
    //check to see if we did our google maps stuff, and that its done
    if (false == googleMapsDone)
    {
        event.preventDefault();

        /* set our submitWaiting flag which we will use in our clalback
           so when we get our google maps response, we post our form right
           away
        */
        submitWaiting = true;

        /* You might want to display a modal or some other kind of notification
           that the form post is 'working' or 'processing' so when the user
           clicks it and doesn't see anything happening, they don't bail
           or click it 800 times out of frustration
        */

        return false;
    }
});
</script>

编辑:我意识到我在下面关于它如何工作的评论......很难理解,所以让我在这里解释一下,然后展示一个替代方案。

  1. 用户填写表单
  2. 用户单击按钮在 google 地图上执行操作(示例是在我知道 GMaps 请求的范围/上下文之前编写的,所以这就是这样做的原因)
  3. 如果用户在 GMap 请求完成之前点击“提交”,我们会取消提交并设置标志submitWaiting
  4. GMaps 请求返回,并执行我们的回调。我们的回调知道如何查找submitWaiting,如果设置为true,它会提交表单

对此的替代方法是,您可以将事件更改为地址输入框的 onChange 事件,而不是要求用户交互 GMaps 请求,或者您可以通过提交按钮/事件完成所有操作,如下所示:

<script>    
$('#form').submit(function(event)
{
    //lets look up our user's address!
    doGoogleMapThing(function()//callback from googlemaps
    {
        //do stuff with your inputs, or whatever

        $('#form').submit();
    });

    event.preventDefault();
    return false;
});
</script>

编辑 注意:以上示例假设您使用的是 jquery,并且您的 google map API 请求是通过 javascript 完成的

如果您的 google map api 请求未使用 google maps javascript 库,这仍然是可能的,只需要您通过本地域上的 php 脚本为 API 制作“代理”脚本。 (浏览器限制)。应该是这样的:

<script>
function doGoogleMapThing(callback_when_done)
{
    $.post("/path/to/proxy/script.php", { data: to, post: to_server }, function(response)
    {
        //check & parse response
        callback_when_done(/* Data needed to populate form */);
    });
}
</script>

注意:这两个示例都假定使用 jquery。因为……嗯……你为什么不呢。

以下是您的确切脚本的实现。我稍微改了一下以使用 jquery,因为它让事情变得不那么痛苦了。

<script type=”text/javascript” 
    src=”http://maps.google.com/maps/api/js?sensor=false”></script>
<script type="text/javascript" 
    src="http://code.jquery.com/jquery-1.6.min.js"></script>

<script type=”text/javascript”>    
function codeAddress(callback)
{
    var address = $('#address').val();

    geocoder.geocode( { 'address': address}, function(results, status)
    {
        if (status == google.maps.GeocoderStatus.OK)
        {
             $('#latitude').val( results[0].geometry.location.latitude );
             $('#longitude').val( results[0].geometry.location.longitude );

             if (typeof callback != 'undefined') //call the callback.
             {
                 callback(true);
             }
        }
    });

    /* Just in case google returns a bad response or doesn't return at all
       we need to add a timeout that will call the callback after 10 seconds
       just so we make sure our user doesn't hang.
    */
    setTimeout(function(){
        callback(false); //pass false indicating no/invalid response
    }, 10000); //10000ms = 10s
}


/* We are using the reallySubmit variable as a flag here, to know when we finish
   our call to google maps and that we want to really submit our form.

   we have to do this because the form.submit() call fires the form's submit event
   again, and we end up going into an infinite loop.

   an alternative to this would be to bind your form processing to the form's submit
   button's click event. that should also pick up any presses of the enter key, also.
   the solution below also works.
*/
var reallySubmit = false;
$('#form').submit(function(event)
{
    if (false == reallySubmit)
    {
        //lets look up our user's address!
        codeAddress(function(success)//callback from googlemaps
        {
            reallySubmit = true;  
            $('#form').submit();
        });

        event.preventDefault();
        return false;
    }
});
</script>

【讨论】:

  • 谢谢,吉姆。但在我看来,提交功能中缺少一些东西。如果 GoogleMaps 的内容比 onSubmit 函数触发的时间长怎么办?那么提交将永远失败。难道不应该有某种监听器或者提交函数中的循环等待 Goggle 完成然后才验证表单吗?
  • 如果这是您想要的行为,您绝对可以这样做。不过,您不需要循环或计时器来完成它,我将编辑我的答案以向您展示一个示例。
  • @Jim - 非常感谢您的更新。你是个读心术。我希望 javascript 获取用户输入的地址,将其发送到 GMaps 并获取 lat/long 作为回复,将它们放在隐藏字段中,然后才提交表单供 PHP 处理。但我仍然看不到你如何处理等待。在我看来,当单击提交按钮并且只触发一次时,这两个函数都被触发了。看起来提交仍然不会发生,但 submitWaiting 标志会更改为 true。那我们不是又被困住了吗?
  • @Ash - 在我的示例代码中,按钮 (#gmap_button) 附加了一个事件。在我的示例中,我假设用户单击某些东西(除了提交)来获取他们的信息。您可以将其附加到地址输入上的change 事件,或者如果需要,甚至可以附加到表单上的submit 事件。但在我的示例中,发生的情况是:用户单击按钮执行 gmap 操作。如果他们单击提交,则提交看到 gmap 未完成,不提交表单,但设置 gmap 函数回调知道要查找的标志,如果存在,则在 gmap 调用完成时提交表单。这有意义吗?
  • @Jim - 是的,当用户单击提交按钮时,我需要这一切发生。所以你最新的例子是最合适的(而且看起来最简单)。如果我理解正确 $('#form').submit() 只会在 googleMaps 内容完成后发生。如果是这样,那么现在我们遇到了您之前描述的问题 - 如果 Google 的东西需要太长时间怎么办?我想等待最多 10 秒,然后在没有完成谷歌的东西的情况下提交。有可能吗?
【解决方案2】:

这根本不可能——PHP 运行在服务器上,JavaScript 运行在客户端。您确实需要使用 Ajax。

【讨论】:

  • 您也可以创建一个隐藏的iframe 来发送请求。虽然不好,但它会工作!
  • 更重要的是,PHP 已经完成运行,并且在任何 Javascript 引擎进入之前,上下文已经不复存在 long
  • 并非根本不可能,根本上被误解了。
  • @Tomalak 和@Jim - 我知道构建 html 的 PHP 在 javascript 运行之前完成运行,但这是构建您正在谈论的 html 的 php 函数。我还在我的问题中提到了另一个 php 函数 - 一个处理提交的表单的函数。我需要这个只有在 javascript 完成工作后才能运行。
【解决方案3】:

您可以有一个隐藏的IFRAME 元素,一旦您的javascript 结束,您就可以在其中加载一个php 文档。简单但有效(只要您的 php 文档来自同一个 URL)。

【讨论】:

    【解决方案4】:

    AJAX 恐怕是必须的。您需要将用户重定向到包含脚本的页面。

    一旦服务器运行 PHP 生成 HTML,然后将其发送到浏览器,就无法再次运行它,或者实际上从服务器运行任何东西,而不向服务器发出请求。这就是服务器端脚本语言的本质,PHP 就是其中之一。

    【讨论】:

    • 虽然,我要补充一点,这远没有听起来那么可怕。潜入水中,我相信你会游泳!
    • 谢谢。但是 - 那么怎么会有处理提交表单的函数呢?当然,这是在表单构建(并提交)之后运行的。
    • 是的,您可以将表单提交到当前页面,并过滤 php 函数以仅在表单已提交时运行 (if(isset($_POST['your_form_field']). ..)
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2021-04-06
    • 2022-10-07
    • 2013-05-05
    相关资源
    最近更新 更多