Code Update

Cleaned up the horribly formatted code into a much more consistant format.

Removed a unused method inside library.php

Added an onion service SSL error suppress option.

Fixed up a couple typos.
monerowp
Alicesland 7 years ago
parent d6438eba52
commit ca16660639

3
.gitignore vendored

@ -0,0 +1,3 @@
*.xml
*.iml

@ -1,18 +1,21 @@
<?php
/* Main Gateway of Monero using a daemon online */
class Monero_Gateway extends WC_Payment_Gateway
{
private $reloadTime = 30000;
private $discount;
private $confirmed = false;
private $monero_daemon;
function __construct()
{
$this->id = "monero_gateway";
$this->method_title = __("Monero GateWay", 'monero_gateway');
$this->method_description = __("Monero Payment Gateway Plug-in for WooCommerce. You can find more information about this payment gateway on our website. You'll need a daemon online for your address.", 'monero_gateway');
$this->title = __("Monero Gateway", 'monero_gateway');
$this->version = "0.2";
$this->version = "0.21";
//
$this->icon = apply_filters('woocommerce_offline_icon', '');
$this->has_fields = false;
@ -41,50 +44,14 @@ class Monero_Gateway extends WC_Payment_Gateway
add_action('woocommerce_thankyou_' . $this->id, array($this, 'instruction'));
if (is_admin()) {
/* Save Settings */
add_action('woocommerce_update_options_payment_gateways_' . $this->id, array($this,'process_admin_options'
));
add_filter( 'woocommerce_currencies', array($this,'add_my_currency') );
add_action('woocommerce_update_options_payment_gateways_' . $this->id, array($this, 'process_admin_options'));
add_filter('woocommerce_currencies', 'add_my_currency');
add_filter('woocommerce_currency_symbol', 'add_my_currency_symbol', 10, 2);
add_action('woocommerce_email_before_order_table', array($this, 'email_instructions'), 10, 2);
}
$this->monero_daemon = new Monero_Library($this->host . ':' . $this->port . '/json_rpc', $this->username, $this->password);
}
public function add_my_currency( $currencies ) {
$currencies['XMR'] = __('Monero','woocommerce');
return $currencies;
}
public function add_my_currency_symbol( $currency_symbol, $currency ) {
switch( $currency ) {
case 'XMR': $currency_symbol = 'XMR'; break;
}
return $currency_symbol;
}
public function admin_options()
{
$this->log->add('Monero_gateway', '[SUCCESS] Monero Settings OK');
echo "<h1>Monero Payment Gateway</h1>";
echo "<p>Welcome to Monero Extension for WooCommerce. Getting started: Make a connection with daemon <a href='https://reddit.com/u/serhack'>Contact Me</a>";
echo "<div style='border:1px solid #DDD;padding:5px 10px;font-weight:bold;color:#223079;background-color:#9ddff3;'>";
$this->getamountinfo();
echo "</div>";
echo "<table class='form-table'>";
$this->generate_settings_html();
echo "</table>";
echo "<h4>Learn more about using a password with the monero wallet-rpc <a href=\"https://github.com/cryptochangements34/monerowp/blob/master/README.md\">here</a></h4>";
}
public function init_form_fields()
{
$this->form_fields = array(
@ -157,81 +124,66 @@ public function add_my_currency_symbol( $currency_symbol, $currency ) {
'description' => __('Check this box if you are using testnet', 'monero_gateway'),
'default' => 'no'
),
'onion_service' => array(
'title' => __(' Onion Service', 'monero_gateway'),
'label' => __('Enable Onion Service', 'monero_gateway'),
'type' => 'checkbox',
'description' => __('Check this box if you are running on an Onion Service (Suppress SSL errors)', 'monero_gateway'),
'default' => 'no'
),
);
}
public function retriveprice($currency)
public function add_my_currency($currencies)
{
$xmr_price = file_get_contents('https://min-api.cryptocompare.com/data/price?fsym=XMR&tsyms=BTC,USD,EUR,CAD,INR,GBP&extraParams=monero_woocommerce');
$price = json_decode($xmr_price, TRUE);
if(!isset($price)){
$this->log->add('Monero_Gateway', '[ERROR] Unable to get the price of Monero');
}
if ($currency == 'USD') {
return $price['USD'];
}
if ($currency == 'EUR') {
return $price['EUR'];
}
if ($currency == 'CAD'){
return $price['CAD'];
}
if ($currency == 'GBP'){
return $price['GBP'];
}
if ($currency == 'INR'){
return $price['INR'];
$currencies['XMR'] = __('Monero', 'woocommerce');
return $currencies;
}
if($currency == 'XMR'){
$price = '1';
return $price;
function add_my_currency_symbol($currency_symbol, $currency)
{
switch ($currency) {
case 'XMR':
$currency_symbol = 'XMR';
break;
}
return $currency_symbol;
}
public function changeto($amount, $currency, $payment_id)
{
global $wpdb;
// This will create a table named whatever the payment id is inside the database "WordPress"
$create_table = "CREATE TABLE IF NOT EXISTS $payment_id (
rate INT
)";
$wpdb->query($create_table);
$rows_num = $wpdb->get_results("SELECT count(*) as count FROM $payment_id");
if($rows_num[0]->count > 0) // Checks if the row has already been created or not
public function admin_options()
{
$stored_rate = $wpdb->get_results("SELECT rate FROM $payment_id");
$this->log->add('Monero_gateway', '[SUCCESS] Monero Settings OK');
$stored_rate_transformed = $stored_rate[0]->rate / 100; //this will turn the stored rate back into a decimaled number
echo "<h1>Monero Payment Gateway</h1>";
if(isset($this->discount))
{
$discount_decimal = $this->discount / 100;
$new_amount = $amount / $stored_rate_transformed;
$discount = $new_amount * $discount_decimal;
$final_amount = $new_amount - $discount;
$rounded_amount = round($final_amount, 12);
}
else{
$new_amount = $amount / $stored_rate_transformed;
$rounded_amount = round($new_amount, 12); //the moneo wallet can't handle decimals smaller than 0.000000000001
}
echo "<p>Welcome to Monero Extension for WooCommerce. Getting started: Make a connection with daemon <a href='https://reddit.com/u/serhack'>Contact Me</a>";
echo "<div style='border:1px solid #DDD;padding:5px 10px;font-weight:bold;color:#223079;background-color:#9ddff3;'>";
$this->getamountinfo();
echo "</div>";
echo "<table class='form-table'>";
$this->generate_settings_html();
echo "</table>";
echo "<h4>Learn more about using a password with the monero wallet-rpc <a href=\"https://github.com/cryptochangements34/monerowp/blob/master/README.md\">here</a></h4>";
}
else // If the row has not been created then the live exchange rate will be grabbed and stored
{
$xmr_live_price = $this->retriveprice($currency);
$live_for_storing = $xmr_live_price * 100; //This will remove the decimal so that it can easily be stored as an integer
$new_amount = $amount / $xmr_live_price;
$rounded_amount = round($new_amount, 12);
$wpdb->query("INSERT INTO $payment_id (rate)
VALUES ($live_for_storing)");
public function getamountinfo()
{
$wallet_amount = $this->monero_daemon->getbalance();
if (!isset($wallet_amount)) {
$this->log->add('Monero_gateway', '[ERROR] No connection with daemon');
$wallet_amount['balance'] = "0";
$wallet_amount['unlocked_balance'] = "0";
}
$real_wallet_amount = $wallet_amount['balance'] / 1000000000000;
$real_amount_rounded = round($real_wallet_amount, 6);
return $rounded_amount;
$unlocked_wallet_amount = $wallet_amount['unlocked_balance'] / 1000000000000;
$unlocked_amount_rounded = round($unlocked_wallet_amount, 6);
echo "Your balance is: " . $real_amount_rounded . " XMR </br>";
echo "Unlocked balance: " . $unlocked_amount_rounded . " XMR </br>";
}
// Submit payment and handle response
public function process_payment($order_id)
{
$order = wc_get_order($order_id);
@ -250,9 +202,8 @@ public function add_my_currency_symbol( $currency_symbol, $currency ) {
}
// Submit payment and handle response
// Validate fields
public function validate_fields()
{
if ($this->check_monero() != TRUE) {
@ -262,6 +213,8 @@ public function add_my_currency_symbol( $currency_symbol, $currency ) {
}
// Validate fields
public function check_monero()
{
$monero_address = $this->settings['monero_address'];
@ -271,17 +224,6 @@ public function add_my_currency_symbol( $currency_symbol, $currency ) {
return false;
}
private function set_paymentid_cookie()
{
if(!isset($_COOKIE['payment_id']))
{
$payment_id = bin2hex(openssl_random_pseudo_bytes(8));
setcookie('payment_id', $payment_id, time()+2700);
}
else
$payment_id = $_COOKIE['payment_id'];
return $payment_id;
}
public function instruction($order_id)
{
$order = wc_get_order($order_id);
@ -304,15 +246,13 @@ public function add_my_currency_symbol( $currency_symbol, $currency ) {
$message = $this->verify_payment($payment_id, $amount_xmr2, $order);
if ($this->confirmed) {
$color = "006400";
}
else{
} else {
$color = "DC143C";
}
echo "<h4><font color=$color>" . $message . "</font></h4>";
echo "<link rel='stylesheet' href='https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css'>";
echo "<div class='row'>
<div class='col-sm-12 col-md-12 col-lg-12'>
<div class='panel panel-default' id='PaymentBox_de3a227fb470475'>
<div class='panel-body'>
@ -329,62 +269,103 @@ public function add_my_currency_symbol( $currency_symbol, $currency ) {
<small>If you need help with how to pay with Monero or want to learn more about it, please go to the Monero<a href='#'>site</a>. </small>
</div>
<div class='col-sm-12 col-md-12 col-lg-12'>
</div>
</div>
</div>
</div>
</div>
</div>
<script type='text/javascript'>
setTimeout(function () { location.reload(true); }, $this->reloadTime);
</script>";
<script type='text/javascript'>setTimeout(function () { location.reload(true); }, $this->reloadTime);</script>";
}
// Check if we are forcing SSL on checkout pages
// Custom function not required by the Gateway
public function do_ssl_check()
private function set_paymentid_cookie()
{
if ($this->enabled == "yes") {
if (get_option('woocommerce_force_ssl_checkout') == "no") {
echo "<div class=\"error\"><p>" . sprintf(__("<strong>%s</strong> is enabled and WooCommerce is not forcing the SSL certificate on your checkout page. Please ensure that you have a valid SSL certificate and that you are <a href=\"%s\">forcing the checkout pages to be secured.</a>"), $this->method_title, admin_url('admin.php?page=wc-settings&tab=checkout')) . "</p></div>";
}
if (!isset($_COOKIE['payment_id'])) {
$payment_id = bin2hex(openssl_random_pseudo_bytes(8));
setcookie('payment_id', $payment_id, time() + 2700);
} else
$payment_id = $_COOKIE['payment_id'];
return $payment_id;
}
public function changeto($amount, $currency, $payment_id)
{
global $wpdb;
// This will create a table named whatever the payment id is inside the database "WordPress"
$create_table = "CREATE TABLE IF NOT EXISTS $payment_id (
rate INT
)";
$wpdb->query($create_table);
$rows_num = $wpdb->get_results("SELECT count(*) as count FROM $payment_id");
if ($rows_num[0]->count > 0) // Checks if the row has already been created or not
{
$stored_rate = $wpdb->get_results("SELECT rate FROM $payment_id");
$stored_rate_transformed = $stored_rate[0]->rate / 100; //this will turn the stored rate back into a decimaled number
if (isset($this->discount)) {
$discount_decimal = $this->discount / 100;
$new_amount = $amount / $stored_rate_transformed;
$discount = $new_amount * $discount_decimal;
$final_amount = $new_amount - $discount;
$rounded_amount = round($final_amount, 12);
} else {
$new_amount = $amount / $stored_rate_transformed;
$rounded_amount = round($new_amount, 12); //the moneo wallet can't handle decimals smaller than 0.000000000001
}
} else // If the row has not been created then the live exchange rate will be grabbed and stored
{
$xmr_live_price = $this->retriveprice($currency);
$live_for_storing = $xmr_live_price * 100; //This will remove the decimal so that it can easily be stored as an integer
$new_amount = $amount / $xmr_live_price;
$rounded_amount = round($new_amount, 12);
public function connect_daemon(){
$host = $this->settings['daemon_host'];
$port = $this->settings['daemon_port'];
$monero_library = new Monero($host, $port);
if( $monero_library->works() == true){
echo "<div class=\"notice notice-success is-dismissible\"><p>Everything works! Congratulations and welcome to Monero. <button type=\"button\" class=\"notice-dismiss\">
<span class=\"screen-reader-text\">Dismiss this notice.</span>
</button></p></div>";
$wpdb->query("INSERT INTO $payment_id (rate)
VALUES ($live_for_storing)");
}
return $rounded_amount;
}
else{
$this->log->add('Monero_gateway','[ERROR] Plugin can not reach wallet rpc.');
echo "<div class=\" notice notice-error\"><p>Error with connection of daemon, see documentation!</p></div>";
} }
// Check if we are forcing SSL on checkout pages
// Custom function not required by the Gateway
public function verify_payment($payment_id, $amount, $order_id){
public function retriveprice($currency)
{
$xmr_price = file_get_contents('https://min-api.cryptocompare.com/data/price?fsym=XMR&tsyms=BTC,USD,EUR,CAD,INR,GBP&extraParams=monero_woocommerce');
$price = json_decode($xmr_price, TRUE);
if (!isset($price)) {
$this->log->add('Monero_Gateway', '[ERROR] Unable to get the price of Monero');
}
switch ($currency) {
case 'USD':
return $price['USD'];
case 'EUR':
return $price['EUR'];
case 'CAD':
return $price['CAD'];
case 'GBP':
return $price['GBP'];
case 'INR':
return $price['INR'];
case 'XMR':
$price = '1';
return $price;
}
}
public function verify_payment($payment_id, $amount, $order_id)
{
/*
* function for verifying payments
* Check if a payment has been made with this payment id then notify the merchant
*/
$message = "We are waiting for your payment to be confirmed";
$amount_atomic_units = $amount * 1000000000000;
$get_payments_method = $this->monero_daemon->get_payments($payment_id);
if(isset($get_payments_method["payments"][0]["amount"]))
{
if($get_payments_method["payments"][0]["amount"] >= $amount_atomic_units)
{
if (isset($get_payments_method["payments"][0]["amount"])) {
if ($get_payments_method["payments"][0]["amount"] >= $amount_atomic_units) {
$message = "Payment has been received and confirmed. Thanks!";
$this->log->add('Monero_gateway', '[SUCCESS] Payment has been recorded. Congratulations!');
$this->confirmed = true;
@ -396,26 +377,31 @@ public function add_my_currency_symbol( $currency_symbol, $currency ) {
$this->reloadTime = 3000000000000; // Greatly increase the reload time as it is no longer needed
}
}
else
return $message;
}
public function do_ssl_check()
{
$message = "We are waiting for your payment to be confirmed";
if ($this->enabled == "yes" && !$this->settings['onion_service']) {
if (get_option('woocommerce_force_ssl_checkout') == "no") {
echo "<div class=\"error\"><p>" . sprintf(__("<strong>%s</strong> is enabled and WooCommerce is not forcing the SSL certificate on your checkout page. Please ensure that you have a valid SSL certificate and that you are <a href=\"%s\">forcing the checkout pages to be secured.</a>"), $this->method_title, admin_url('admin.php?page=wc-settings&tab=checkout')) . "</p></div>";
}
return $message;
}
public function getamountinfo(){
$wallet_amount = $this->monero_daemon->getbalance();
if(!isset($wallet_amount)){
$this->log->add('Monero_gateway','[ERROR] No connection with daemon');
$wallet_amount['balance'] = "0";
$wallet_amount['unlocked_balance'] = "0";
}
$real_wallet_amount = $wallet_amount['balance'] / 1000000000000;
$real_amount_rounded = round($real_wallet_amount, 6);
$unlocked_wallet_amount = $wallet_amount['unlocked_balance'] / 1000000000000;
$unlocked_amount_rounded = round($unlocked_wallet_amount, 6);
public function connect_daemon()
{
$host = $this->settings['daemon_host'];
$port = $this->settings['daemon_port'];
$monero_library = new Monero($host, $port);
if ($monero_library->works() == true) {
echo "<div class=\"notice notice-success is-dismissible\"><p>Everything works! Congratulations and welcome to Monero. <button type=\"button\" class=\"notice-dismiss\">
<span class=\"screen-reader-text\">Dismiss this notice.</span>
</button></p></div>";
echo "Your balance is: ".$real_amount_rounded. " XMR </br>";
echo "Unlocked balance: ".$unlocked_amount_rounded." XMR </br>";
} else {
$this->log->add('Monero_gateway', '[ERROR] Plugin can not reach wallet rpc.');
echo "<div class=\" notice notice-error\"><p>Error with connection of daemon, see documentation!</p></div>";
}
}
}

@ -1,4 +1,5 @@
<?php
/**
* library.php
*
@ -12,14 +13,12 @@
class Monero_Library
{
protected $url = null, $is_debug = false, $parameters_structure = 'array';
private $username;
private $password;
protected $curl_options = array(
CURLOPT_CONNECTTIMEOUT => 8,
CURLOPT_TIMEOUT => 8
);
private $username;
private $password;
private $httpErrors = array(
400 => '400 Bad Request',
401 => '401 Unauthorized',
@ -43,9 +42,11 @@ class Monero_Library
$this->password = $pPass;
}
private function getHttpErrorMessage($pErrorNumber)
public function validate($pFailed, $pErrMsg)
{
return isset($this->httpErrors[$pErrorNumber]) ? $this->httpErrors[$pErrorNumber] : null;
if ($pFailed) {
echo $pErrMsg;
}
}
public function setDebug($pIsDebug)
@ -69,17 +70,32 @@ class Monero_Library
public function setCurlOptions($pOptionsArray)
{
if (is_array($pOptionsArray))
{
if (is_array($pOptionsArray)) {
$this->curl_options = $pOptionsArray + $this->curl_options;
}
else
{
} else {
echo 'Invalid options type.';
}
return $this;
}
public function _print($json)
{
$json_encoded = json_encode($json, JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES);
echo $json_encoded;
}
public function address()
{
$address = $this->_run('getaddress');
return $address;
}
public function _run($method, $params = null)
{
$result = $this->request($method, $params);
return $result; //the result is returned as an array
}
private function request($pMethod, $pParams)
{
static $requestId = 0;
@ -106,24 +122,46 @@ class Monero_Library
// check if response is correct
$this->validate(empty($responseDecoded['id']), 'Invalid response data structure: ' . $responseMessage);
$this->validate($responseDecoded['id'] != $requestId, 'Request id: ' . $requestId . ' is different from Response id: ' . $responseDecoded['id']);
if (isset($responseDecoded['error']))
{
if (isset($responseDecoded['error'])) {
$errorMessage = 'Request have return error: ' . $responseDecoded['error']['message'] . '; ' . "\n" .
'Request: ' . $request . '; ';
if (isset($responseDecoded['error']['data']))
{
if (isset($responseDecoded['error']['data'])) {
$errorMessage .= "\n" . 'Error data: ' . $responseDecoded['error']['data'];
}
$this->validate(!is_null($responseDecoded['error']), $errorMessage);
}
return $responseDecoded['result'];
}
protected function debug($pAdd, $pShow = false)
{
static $debug, $startTime;
// is_debug off return
if (false === $this->is_debug) {
return;
}
// add
$debug .= $pAdd;
// get starttime
$startTime = empty($startTime) ? array_sum(explode(' ', microtime())) : $startTime;
if (true === $pShow and !empty($debug)) {
// get endtime
$endTime = array_sum(explode(' ', microtime()));
// performance summary
$debug .= 'Request time: ' . round($endTime - $startTime, 3) . ' s Memory usage: ' . round(memory_get_usage() / 1024) . " kb\r\n";
echo nl2br($debug);
// send output immediately
flush();
// clean static
$debug = $startTime = null;
}
}
protected function & getResponse(&$pRequest)
{
// do the actual connection
$ch = curl_init();
if ( !$ch)
{
if (!$ch) {
throw new RuntimeException('Could\'t initialize a cURL session');
}
curl_setopt($ch, CURLOPT_URL, $this->url);
@ -136,21 +174,18 @@ class Monero_Library
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
if ( !curl_setopt_array($ch, $this->curl_options))
{
if (!curl_setopt_array($ch, $this->curl_options)) {
throw new RuntimeException('Error while setting curl options');
}
// send the request
$response = curl_exec($ch);
// check http status code
$httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
if (isset($this->httpErrors[$httpCode]))
{
if (isset($this->httpErrors[$httpCode])) {
echo 'Response Http Error - ' . $this->httpErrors[$httpCode];
}
// check for curl error
if (0 < curl_errno($ch))
{
if (0 < curl_errno($ch)) {
echo 'Unable to connect to ' . $this->url . ' Error: ' . curl_error($ch);
}
// close the connection
@ -158,44 +193,11 @@ class Monero_Library
return $response;
}
public function validate($pFailed, $pErrMsg)
{
if ($pFailed)
{
echo $pErrMsg;
}
}
protected function debug($pAdd, $pShow = false)
{
static $debug, $startTime;
// is_debug off return
if (false === $this->is_debug)
{
return;
}
// add
$debug .= $pAdd;
// get starttime
$startTime = empty($startTime) ? array_sum(explode(' ', microtime())) : $startTime;
if (true === $pShow and !empty($debug))
{
// get endtime
$endTime = array_sum(explode(' ', microtime()));
// performance summary
$debug .= 'Request time: ' . round($endTime - $startTime, 3) . ' s Memory usage: ' . round(memory_get_usage() / 1024) . " kb\r\n";
echo nl2br($debug);
// send output imidiately
flush();
// clean static
$debug = $startTime = null;
}
}
//prints result as json
function getJsonLastErrorMsg()
{
if (!function_exists('json_last_error_msg'))
{
if (!function_exists('json_last_error_msg')) {
function json_last_error_msg()
{
static $errors = array(
@ -212,41 +214,19 @@ class Monero_Library
}
// Fix PHP 5.2 error caused by missing json_last_error function
if (function_exists('json_last_error'))
{
if (function_exists('json_last_error')) {
return json_last_error() ? json_last_error_msg() : null;
}
else
{
} else {
return null;
}
}
public function _run($method,$params = null)
{
$result = $this->request($method, $params);
return $result; //the result is returned as an array
}
//prints result as json
public function _print($json)
{
$json_encoded = json_encode($json, JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES);
echo $json_encoded;
}
/*
* The following functions can all be called to interact with the monero rpc wallet
* They will majority of them will return the result as an array
* Example: $daemon->address(); where $daemon is an instance of this class, will return the wallet address as string within an array
*/
public function address()
{
$address = $this->_run('getaddress');
return $address;
}
public function getbalance()
{
$balance = $this->_run('getbalance');
@ -280,8 +260,6 @@ class Monero_Library
return $query_key_method;
}
/* A payment id can be passed as a string
A random payment id will be generatd if one is not given */
public function make_integrated_address($payment_id)
{
$integrate_address_parameters = array('payment_id' => $payment_id);
@ -289,12 +267,14 @@ class Monero_Library
return $integrate_address_method;
}
/* A payment id can be passed as a string
A random payment id will be generatd if one is not given */
public function split_integrated_address($integrated_address)
{
if (!isset($integrated_address)) {
echo "Error: Integrated_Address mustn't be null";
}
else{
} else {
$split_params = array('integrated_address' => $integrated_address);
$split_methods = $this->_run('split_integrated_address', $split_params);
return $split_methods;

@ -12,7 +12,8 @@ if ( ! defined( 'ABSPATH' ) ) {
}
// Include our Gateway Class and register Payment Gateway with WooCommerce
add_action('plugins_loaded', 'monero_init', 0);
function monero_init() {
function monero_init()
{
/* If the class doesn't exist (== WooCommerce isn't installed), return NULL */
if (!class_exists('WC_Payment_Gateway')) return;
@ -23,13 +24,11 @@ function monero_init() {
// Lets add it too WooCommerce
add_filter('woocommerce_payment_gateways', 'monero_gateway');
function monero_gateway( $methods ) {
function monero_gateway($methods)
{
$methods[] = 'Monero_Gateway';
return $methods;
}
}
/*
@ -37,15 +36,18 @@ function monero_init() {
* The url will be http://yourworpress/wp-admin/admin.php?=wc-settings&tab=checkout
*/
add_filter('plugin_action_links_' . plugin_basename(__FILE__), 'monero_payment');
function monero_payment( $links ) {
function monero_payment($links)
{
$plugin_links = array(
'<a href="' . admin_url('admin.php?page=wc-settings&tab=checkout') . '">' . __('Settings', 'monero_payment') . '</a>',
);
return array_merge($plugin_links, $links);
}
add_action('admin_menu', 'monero_create_menu');
function monero_create_menu(){
function monero_create_menu()
{
add_menu_page(
__('Monero', 'textdomain'),
'Monero',

Loading…
Cancel
Save