প্রতিটি অ্যাজাক্স অনুরোধের জন্য কীভাবে একটি অনন্য নোকস পাবেন?


11

পরবর্তী অ্যাজাক্স অনুরোধগুলির জন্য ওয়ার্ডপ্রেসকে একটি অনন্য নকল পুনরুত্পাদন করার বিষয়ে আমি বেশ কয়েকবার আলোচনা দেখেছি, তবে আমার জীবনের জন্য আমি আসলে ওয়ার্ডপ্রেসকে এটি করতে পারি না - প্রতিবার আমি যা অনুরোধ করি তা নতুন হওয়া উচিত বাজে, আমি ওয়ার্ডপ্রেস থেকে একই ননস ফিরে পেয়েছি। আমি ডাব্লুপি'র ননসাইফ লাইফ এবং এমনকি এটি অন্য কোনও কিছুর ধারণাটি বুঝতে পেরেছি, তবে এটি আমার সাহায্য করেনি।

আমি স্থানীয়করণের মাধ্যমে শিরোনামের জেএস বস্তুটিতে নোটস উত্পন্ন করি না - আমি এটি আমার প্রদর্শন পৃষ্ঠায় করি। আমি আমার পৃষ্ঠাটি অ্যাজাক্স অনুরোধটি প্রক্রিয়া করার জন্য পেতে পারি, কিন্তু আমি কলব্যাকের সময় ডাব্লুপি থেকে নতুন ননসের জন্য অনুরোধ করলে আমি একই নাকটি ফিরে পাই এবং আমি কী ভুল করছি তা আমি জানি না ... শেষ পর্যন্ত আমি চাই এটি প্রসারিত করুন যাতে পৃষ্ঠায় একাধিক আইটেম থাকতে পারে, প্রতিটি যুক্ত করার / অপসারণের ক্ষমতা সহ - তাই আমার একটি সমাধান দরকার যা এক পৃষ্ঠায় একাধিক পরবর্তী অ্যাজাক্স অনুরোধকে মঞ্জুরি দেয়।

(এবং আমার বলা উচিত যে আমি এই সমস্ত কার্যকারিতা প্লাগইনে রেখেছি, সুতরাং সম্মুখ-প্রান্তের "প্রদর্শন পৃষ্ঠা" আসলে প্লাগইনের সাথে অন্তর্ভুক্ত একটি ফাংশন ...)

ফাংশন.এফপি: স্থানীয়করণ করুন, তবে আমি এখানে একটি ননস তৈরি করি না

wp_localize_script('myjs', 'ajaxVars', array('ajaxurl' => 'admin-ajax.php')));

ফোন করে জেএস:

$("#myelement").click(function(e) {
    e.preventDefault();
    post_id = $(this).data("data-post-id");
    user_id = $(this).data("data-user-id");
    nonce = $(this).data("data-nonce");
    $.ajax({
      type: "POST",
      dataType: "json",
      url: ajaxVars.ajaxurl,
      data: {
         action: "myfaves",
         post_id: post_id,
         user_id: user_id,
         nonce: nonce
      },
      success: function(response) {
         if(response.type == "success") {
            nonce = response.newNonce;
            ... other stuff
         }
      }
  });
});

পিএইচপি গ্রহণ:

function myFaves() {
   $ajaxNonce = 'myplugin_myaction_nonce_' . $postID;
   if (!wp_verify_nonce($_POST['nonce'], $ajaxNonce))
      exit('Sorry!');

   // Get various POST vars and do some other stuff...

   // Prep JSON response & generate new, unique nonce
   $newNonce = wp_create_nonce('myplugin_myaction_nonce_' . $postID . '_' 
       . str_replace('.', '', gettimeofday(true)));
   $response['newNonce'] = $newNonce;

   // Also let the page process itself if there is no JS/Ajax capability
   } else {
      header("Location: " . $_SERVER["HTTP_REFERER"];
   }
   die();
}

সামনের পিএইচপি প্রদর্শন ফাংশন, যার মধ্যে রয়েছে:

$nonce = wp_create_nonce('myplugin_myaction_nonce_' . $post->ID);
$link = admin_url('admin-ajax.php?action=myfaves&post_id=' . $post->ID
   . '&user_id=' . $user_ID
   . '&nonce=' . $nonce);

echo '<a id="myelement" data-post-id="' . $post->ID
   . '" data-user-id="' . $user_ID
   . '" data-nonce="' . $nonce
   . '" href="' . $link . '">My Link</a>';

এই মুহুর্তে আমি WP প্রতিটি নতুন Ajax অনুরোধের জন্য একটি অনন্য ননকে পুনরুত্পাদন করার জন্য কোনও ক্লু বা পয়েন্টারের জন্য সত্যই কৃতজ্ঞ থাকব ...


আপডেট: আমি আমার সমস্যার সমাধান করেছি। উপরের কোড স্নিপেটগুলি বৈধ, তবে আমি পিএইচপি কলব্যাকে on নতুননস ক্রিয়েশনটি মাইক্রোসেকেন্ডস স্ট্রিং সংযোজন করতে পেরেছি তা পরবর্তী অ্যাজাক্স অনুরোধে অনন্য কিনা তা নিশ্চিত করতে।


A থেকে খুব সংক্ষিপ্ত বর্ণন: আপনি আপাতত তৈরি করছেন পরে আপনি এটি পেয়েছি (প্রদর্শনের)? লোকালাইজ কল চলাকালীন কেন আপনি এটি তৈরি করছেন না?
কায়সার

JQuery একটি # মাইলেমেন্ট লিঙ্কে "ডেটা-ননস" বৈশিষ্ট্যটি থেকে প্রাথমিক ননস ব্যবহার করছে এবং ধারণাটি হ'ল পৃষ্ঠাটি উভয়ই অ্যাজাক্স বা নিজে থেকেই প্রক্রিয়াভুক্ত হতে পারে। আমার কাছে মনে হয়েছিল যে একবার লোকালাইজ কলের মাধ্যমে নোসেস তৈরি করা এটি নন-জেএস প্রসেসিং থেকে বাদ দেবে, তবে আমি এ সম্পর্কে ভুল হতে পারি। যেভাবেই হোক ওয়ার্ডপ্রেস আমাকে একই অলস ফিরিয়ে দেয় ...
টিম

এছাড়াও: লোকালাইজ কলটিতে নোকসটি রাখলে এমন কোনও পৃষ্ঠাতে একাধিক আইটেম থাকা আটকাতে পারে না যেখানে প্রতিটি আইটেমের অ্যাজাক্স অনুরোধের জন্য অনন্য নোট থাকতে পারে?
টিম

লোকালাইজের ভিতরে ননস তৈরি করা সেই স্ক্রিপ্টের জন্য তৈরি করে এটি উপলব্ধ করত। তবে আপনি সীমাহীন পরিমাণে অন্যান্য (কী নামযুক্ত) পৃথক নোকসের সাহায্যে স্থানীয়করণের মানগুলিও যুক্ত করতে পারেন।
কায়সার

যদি আপনি এটি সমাধান করে থাকেন তবে আপনাকে আপনার উত্তর পোস্ট করতে এবং এটি "স্বীকৃত" হিসাবে চিহ্নিত করতে উত্সাহিত করা হবে। এটি সাইটটি সুসংহত রাখতে সহায়তা করবে। আমি কেবল আপনার কোডটি নিয়ে গণ্ডগোল করছি এবং বেশ কয়েকটি জিনিস আমার পক্ষে কাজ করে না, তাই আপনার সমাধান পোস্ট করার জন্য অনুরোধটি দ্বিগুণ করুন।
s_ha_dum

উত্তর:


6

এখানে আমার নিজের প্রশ্নের একটি দীর্ঘ উত্তর দেওয়া হয়েছে যা পরবর্তী অ্যাজাক্স অনুরোধগুলির জন্য অনন্য নোট তৈরির প্রশ্নটি সম্বোধনের বাইরে। এটি একটি "ফেভারিটে অ্যাড করুন" বৈশিষ্ট্য যা উত্তরের উদ্দেশ্যে জেনেরিক করা হয়েছিল (আমার বৈশিষ্ট্যটি ব্যবহারকারীদের পছন্দসই তালিকায় ফটো সংযুক্তির পোস্ট আইডি যুক্ত করতে দেয় তবে এটি নির্ভর করে এমন অন্যান্য বৈশিষ্ট্যগুলির ক্ষেত্রেও প্রযোজ্য হতে পারে আয়াক্স)। আমি এটিকে একটি স্বতন্ত্র প্লাগইন হিসাবে কোড করে দিয়েছি এবং কয়েকটি আইটেম অনুপস্থিত রয়েছে - তবে আপনি যদি বৈশিষ্ট্যটির অনুলিপি করতে চান তবে এই সংক্ষিপ্তসার সরবরাহ করার জন্য যথেষ্ট বিশদ হওয়া উচিত। এটি স্বতন্ত্র পোস্ট / পৃষ্ঠায় কাজ করবে তবে এটি পোস্টের তালিকায়ও কাজ করবে (যেমন আপনি অ্যাজাক্সের মাধ্যমে পছন্দসই ইনলাইনগুলিতে আইটেমগুলি যুক্ত / সরাতে পারেন এবং প্রতিটি পোস্টের প্রতিটি অ্যাজাক্স অনুরোধের জন্য নিজস্ব অনন্য ননস থাকবে)। মনে রাখবেন যে সেখানে '

scripts.php

/**
* Enqueue front-end jQuery
*/
function enqueueFavoritesJS()
{
    // Only show Favorites Ajax JS if user is logged in
    if (is_user_logged_in()) {
        wp_enqueue_script('favorites-js', MYPLUGIN_BASE_URL . 'js/favorites.js', array('jquery'));
        wp_localize_script('favorites-js', 'ajaxVars', array('ajaxurl' => admin_url('admin-ajax.php')));
    }
}
add_action('wp_enqueue_scripts', 'enqueueFavoritesJS');

প্রিয়. js (প্রচুর পরিমাণে ডিবাগ সামগ্রী যা সরানো যায়)

$(document).ready(function()
{
    // Toggle item in Favorites
    $(".faves-link").click(function(e) {
        // Prevent self eval of requests and use Ajax instead
        e.preventDefault();
        var $this = $(this);
        console.log("Starting click event...");

        // Fetch initial variables from the page
        post_id = $this.attr("data-post-id");
        user_id = $this.attr("data-user-id");
        the_toggle = $this.attr("data-toggle");
        ajax_nonce = $this.attr("data-nonce");

        console.log("data-post-id: " + post_id);
        console.log("data-user-id: " + user_id);
        console.log("data-toggle: " + the_toggle);
        console.log("data-nonce: " + ajax_nonce);
        console.log("Starting Ajax...");

        $.ajax({
            type: "POST",
            dataType: "json",
            url: ajaxVars.ajaxurl,
            data: {
                // Send JSON back to PHP for eval
                action : "myFavorites",
                post_id: post_id,
                user_id: user_id,
                _ajax_nonce: ajax_nonce,
                the_toggle: the_toggle
            },
            beforeSend: function() {
                if (the_toggle == "y") {
                    $this.text("Removing from Favorites...");
                    console.log("Removing...");
                } else {
                    $this.text("Adding to Favorites...");
                    console.log("Adding...");
                }
            },
            success: function(response) {
                // Process JSON sent from PHP
                if(response.type == "success") {
                    console.log("Success!");
                    console.log("New nonce: " + response.newNonce);
                    console.log("New toggle: " + response.theToggle);
                    console.log("Message from PHP: " + response.message);
                    $this.text(response.message);
                    $this.attr("data-toggle", response.theToggle);
                    // Set new nonce
                    _ajax_nonce = response.newNonce;
                    console.log("_ajax_nonce is now: " + _ajax_nonce);
                } else {
                    console.log("Failed!");
                    console.log("New nonce: " + response.newNonce);
                    console.log("Message from PHP: " + response.message);
                    $this.parent().html("<p>" + response.message + "</p>");
                    _ajax_nonce = response.newNonce;
                    console.log("_ajax_nonce is now: " + _ajax_nonce);
                }
            },
            error: function(e, x, settings, exception) {
                // Generic debugging
                var errorMessage;
                var statusErrorMap = {
                    '400' : "Server understood request but request content was invalid.",
                    '401' : "Unauthorized access.",
                    '403' : "Forbidden resource can't be accessed.",
                    '500' : "Internal Server Error",
                    '503' : "Service Unavailable"
                };
                if (x.status) {
                    errorMessage = statusErrorMap[x.status];
                    if (!errorMessage) {
                        errorMessage = "Unknown Error.";
                    } else if (exception == 'parsererror') {
                        errorMessage = "Error. Parsing JSON request failed.";
                    } else if (exception == 'timeout') {
                        errorMessage = "Request timed out.";
                    } else if (exception == 'abort') {
                        errorMessage = "Request was aborted by server.";
                    } else {
                        errorMessage = "Unknown Error.";
                    }
                    $this.parent().html(errorMessage);
                    console.log("Error message is: " + errorMessage);
                } else {
                    console.log("ERROR!!");
                    console.log(e);
                }
            }
        }); // Close $.ajax
    }); // End click event
});

ফাংশন (সম্মুখ-প্রদর্শন প্রদর্শন এবং অ্যাজাক্স ক্রিয়া)

অ্যাড / অপসারণ পছন্দসই লিঙ্কটি আউটপুট করতে, কেবল এটির মাধ্যমে আপনার পৃষ্ঠা / পোস্টে কল করুন:

if (function_exists('myFavoritesLink') {
    myFavoritesLink($user_ID, $post->ID);
}

ফ্রন্ট-এন্ড ডিসপ্লে ফাংশন:

function myFavoritesLink($user_ID, $postID)
{
    global $user_ID;
    if (is_user_logged_in()) {
        // Set initial element toggle value & link text - udpated by callback
        $myUserMeta = get_user_meta($user_ID, 'myMetadata', true);
        if (is_array($myUserMeta['metadata']) && in_array($postID, $myUserMeta['metadata'])) {
            $toggle = "y";
            $linkText = "Remove from Favorites";
        } else {
            $toggle = "n";
            $linkText = "Add to Favorites";
        }

        // Create Ajax-only nonce for initial request only
        // New nonce returned in callback
        $ajaxNonce = wp_create_nonce('myplugin_myaction_' . $postID);
        echo '<p class="faves-action"><a class="faves-link"' 
            . ' data-post-id="' . $postID 
            . '" data-user-id="' . $user_ID  
            . '" data-toggle="' . $toggle 
            . '" data-nonce="' . $ajaxNonce 
            . '" href="#">' . $linkText . '</a></p>' . "\n";

    } else {
        // User not logged in
        echo '<p>Sign in to use the Favorites feature.</p>' . "\n";
    }

}

অ্যাজাক্স অ্যাকশন ফাংশন:

/**
* Toggle add/remove for Favorites
*/
function toggleFavorites()
{
    if (is_user_logged_in()) {
        // Verify nonce
        $ajaxNonce = 'myplugin_myaction' . $_POST['post_id'];
        if (! wp_verify_nonce($_POST['_ajax_nonce'], $ajaxNonce)) {
            exit('Sorry!');
        }
        // Process POST vars
        if (isset($_POST['post_id']) && is_numeric($_POST['post_id'])) {
            $postID = $_POST['post_id'];
        } else {
            return;
        }
        if (isset($_POST['user_id']) && is_numeric($_POST['user_id'])) {
            $userID = $_POST['user_id'];
        } else {
            return;
        }
        if (isset($_POST['the_toggle']) && ($_POST['the_toggle'] === "y" || $_POST['the_toggle'] === "n")) {
            $toggle = $_POST['the_toggle'];
        } else {
            return;
        }

        $myUserMeta = get_user_meta($userID, 'myMetadata', true);

        // Init myUserMeta array if it doesn't exist
        if ($myUserMeta['myMetadata'] === '' || ! is_array($myUserMeta['myMetadata'])) {
            $myUserMeta['myMetadata'] = array();
        }

        // Toggle the item in the Favorites list
        if ($toggle === "y" && in_array($postID, $myUserMeta['myMetadata'])) {
            // Remove item from Favorites list
            $myUserMeta['myMetadata'] = array_flip($myUserMeta['myMetadata']);
            unset($myUserMeta['myMetadata'][$postID]);
            $myUserMeta['myMetadata'] = array_flip($myUserMeta['myMetadata']);
            $myUserMeta['myMetadata'] = array_values($myUserMeta['myMetadata']);
            $newToggle = "n";
            $message = "Add to Favorites";
        } else {
            // Add item to Favorites list
            $myUserMeta['myMetadata'][] = $postID;
            $newToggle = "y";
            $message = "Remove from Favorites";
        }

        // Prep for the response
        // Nonce for next request - unique with microtime string appended
        $newNonce = wp_create_nonce('myplugin_myaction_' . $postID . '_' 
            . str_replace('.', '', gettimeofday(true)));
        $updateUserMeta = update_user_meta($userID, 'myMetadata', $myUserMeta);

        // Response to jQuery
        if($updateUserMeta === false) {
            $response['type'] = "error";
            $response['theToggle'] = $toggle;
            $response['message'] = "Your Favorites could not be updated.";
            $response['newNonce'] = $newNonce;
        } else {
            $response['type'] = "success";
            $response['theToggle'] = $newToggle;
            $response['message'] = $message;
            $response['newNonce'] = $newNonce;
        }

        // Process with Ajax, otherwise process with self
        if (! empty($_SERVER['HTTP_X_REQUESTED_WITH']) && 
            strtolower($_SERVER['HTTP_X_REQUESTED_WITH']) == 'xmlhttprequest') {
                $response = json_encode($response);
                echo $response;
        } else {
            header("Location: " . $_SERVER["HTTP_REFERER"]);
        }
        exit();
    } // End is_user_logged_in()
}
add_action('wp_ajax_myFavorites', 'toggleFavorites');

3

প্রতিটি এজাক্স অনুরোধের জন্য একটি নতুন ননড পাওয়ার পিছনে যুক্তিটি আমাকে সত্যিই প্রশ্ন করতে হবে। আসল ননসের মেয়াদ শেষ হয়ে যাবে তবে এটি না হওয়া পর্যন্ত এটি একাধিকবার ব্যবহার করা যেতে পারে। জাভাস্ক্রিপ্টটি অ্যাজ্যাক্সের মাধ্যমে এটি গ্রহণের কারণে উদ্দেশ্যটি হেরে যায়, বিশেষত এটি একটি ত্রুটির ক্ষেত্রে সরবরাহ করে। (সময়সীমার মধ্যে কোনও ব্যবহারকারীর সাথে কোনও ক্রিয়াকলাপ সংযুক্ত করার জন্য ননসেসের উদ্দেশ্যটি সামান্য সুরক্ষা হওয়া))

আমার অন্যান্য উত্তর উল্লেখ করার কথা নয়, তবে আমি নতুন এবং উপরে মন্তব্য করতে পারছি না, সুতরাং পোস্ট করা "সমাধান" এর ক্ষেত্রে আপনি প্রতিবারই একটি নতুন আভিজাত্য পাচ্ছেন কিন্তু অনুরোধে এটি ব্যবহার করছেন না। প্রতিটি নতুন ননকে সেইভাবে তৈরি করার সাথে প্রতিবার মাইক্রোসেকেন্ডগুলি একইভাবে পাওয়া শক্ত হবে। পিএইচপি কোডটি মূল ননসের বিরুদ্ধে যাচাই করছে এবং জাভাস্ক্রিপ্টটি মূল ননস সরবরাহ করছে ... সুতরাং এটি কাজ করে (কারণ এটি এখনও শেষ হয়নি)।


1
সমস্যাটি হচ্ছে, ননস ব্যবহারের পরে এটির মেয়াদ শেষ হয়ে যায় এবং প্রতিবারের পরে এজাক্স ফাংশনে -1 আসবে। আপনি যদি পিএইচপি-তে কোনও ফর্মের অংশগুলি বৈধতা দিচ্ছেন এবং মুদ্রণের জন্য ত্রুটিগুলি ফিরিয়ে দেন তবে এটি সমস্যা। ফর্ম ননস ব্যবহার করা হয়েছিল, তবে ক্ষেত্রগুলির পিএইচপি বৈধকরণে ত্রুটি আসলেই ঘটেছে, এবং ফর্মটি আবার জমা দেওয়ার পরে, এটি যাচাই করতে সক্ষম হয় না এবং check_ajax_refererফেরত -১ দেয়, যা আমরা চাই না!
সলোমন ক্লসন
আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.