Add donate and memberships

This commit is contained in:
Joey Kimsey
2024-12-05 15:33:25 -05:00
parent 03aedc3020
commit 402182714e
39 changed files with 2109 additions and 21 deletions

View File

@ -0,0 +1,50 @@
<?php
/**
* app/plugins/members/controllers/admin/members/blog.php
*
* This is the Membership Invoices admin controller.
*
* @package TP Members
* @version 3.0
* @author Joey Kimsey <Joey@thetempusproject.com>
* @link https://TheTempusProject.com
* @license https://opensource.org/licenses/MIT [MIT LICENSE]
*/
namespace TheTempusProject\Controllers\Admin;
use TheTempusProject\Houdini\Classes\Views;
use TheTempusProject\Houdini\Classes\Navigation;
use TheTempusProject\Houdini\Classes\Components;
use TheTempusProject\Classes\AdminController;
use TheTempusProject\Models\MembershipRecords as MemberModel;
class Invoices extends AdminController {
public static $memberships;
public function __construct() {
parent::__construct();
self::$title = 'Admin - Memberships';
self::$memberships = new MemberModel;
$view = Navigation::activePageSelect( 'nav.admin', '/admin/member' );
Components::set( 'ADMINNAV', $view );
}
public function index( $data = null ) {
Views::view( 'members.admin.list', self::$memberships->list() );
}
public function create( $data = null ) {
}
public function edit( $data = null ) {
}
public function view( $data = null ) {
}
public function delete( $data = null ) {
}
public function preview( $data = null ) {
}
}

View File

@ -0,0 +1,40 @@
<?php
/**
* app/plugins/blog/controllers/admin/blog.php
*
* This is the Blog admin controller.
*
* @package TP Blog
* @version 3.0
* @author Joey Kimsey <Joey@thetempusproject.com>
* @link https://TheTempusProject.com
* @license https://opensource.org/licenses/MIT [MIT LICENSE]
*/
namespace TheTempusProject\Controllers\Admin;
use TheTempusProject\Houdini\Classes\Views;
use TheTempusProject\Houdini\Classes\Navigation;
use TheTempusProject\Houdini\Classes\Components;
use TheTempusProject\Classes\AdminController;
use TheTempusProject\Plugins\Members as MemberModel;
use TheTempusProject\Houdini\Classes\Issues;
use TheTempusProject\Bedrock\Functions\Input;
class Members extends AdminController {
public function __construct() {
parent::__construct();
$view = Navigation::activePageSelect( 'nav.admin', '/admin/member' );
Components::set( 'ADMINNAV', $view );
}
public function index( $data = null ) {
self::$title = 'Admin - Membership Webhooks';
if ( !Input::exists( 'submit' ) ) {
return Views::view( 'members.admin.webhooks' );
}
MemberModel::webhookSetup();
Issues::add( 'success', 'Webhooks Generated' );
Issues::add( 'error', 'Now, LEAVE!' );
}
}

View File

@ -0,0 +1,97 @@
<?php
/**
* app/plugins/members/controllers/admin/members/products.php
*
* This is the Membership Products admin controller.
*
* @package TP Members
* @version 3.0
* @author Joey Kimsey <Joey@thetempusproject.com>
* @link https://TheTempusProject.com
* @license https://opensource.org/licenses/MIT [MIT LICENSE]
*/
namespace TheTempusProject\Controllers\Admin;
use TheTempusProject\Houdini\Classes\Views;
use TheTempusProject\Houdini\Classes\Navigation;
use TheTempusProject\Houdini\Classes\Components;
use TheTempusProject\Classes\AdminController;
use TheTempusProject\Models\MembershipProducts;
use TheTempusProject\Bedrock\Functions\Input;
use TheTempusProject\Classes\Forms;
use TheTempusProject\Houdini\Classes\Issues;
use TheTempusProject\Bedrock\Functions\Check;
class Products extends AdminController {
public static $products;
public function __construct() {
parent::__construct();
self::$title = 'Admin - Membership Products';
self::$products = new MembershipProducts;
$view = Navigation::activePageSelect( 'nav.admin', '/admin/products' );
Components::set( 'ADMINNAV', $view );
}
public function index( $data = null ) {
Views::view( 'members.admin.products.list', self::$products->list() );
}
public function create( $data = null ) {
if ( !Input::exists( 'submit' ) ) {
return Views::view( 'members.admin.products.create' );
}
if ( !Forms::check( 'newMembershipProduct' ) ) {
Issues::add( 'error', [ 'There was an error with your request.' => Check::userErrors() ] );
return $this->index();
}
$result = self::$products->create( Input::post( 'name' ), Input::post( 'description' ), Input::post( 'monthly_price' ), Input::post( 'yearly_price' ) );
if ( $result ) {
Issues::add( 'success', 'Your product has been created.' );
return $this->index();
} else {
Issues::add( 'error', [ 'There was an unknown error submitting your data.' => Check::userErrors() ] );
return $this->index();
}
}
public function edit( $id = null ) {
if ( !Input::exists( 'submit' ) ) {
return Views::view( 'members.admin.products.edit', self::$posts->findById( $id ) );
}
if ( !Forms::check( 'editMembershipProduct' ) ) {
Issues::add( 'error', [ 'There was an error with your form.' => Check::userErrors() ] );
return $this->index();
}
if ( self::$posts->updatePost( $id, Input::post( 'title' ), Input::post( 'blogPost' ), Input::post( 'submit' ) ) === true ) {
Issues::add( 'success', 'Post Updated.' );
return $this->index();
}
Issues::add( 'error', 'There was an error with your request.' );
$this->index();
}
public function view( $id = null ) {
$data = self::$products->findById( $id );
if ( $data !== false ) {
return Views::view( 'blog.admin.view', $data );
}
Issues::add( 'error', 'Post not found.' );
$this->index();
}
public function delete( $data = null ) {
if ( $data == null ) {
if ( Input::exists( 'MP_' ) ) {
$data = Input::post( 'MP_' );
}
}
if ( !self::$products->delete( (array) $data ) ) {
Issues::add( 'error', 'There was an error with your request.' );
} else {
Issues::add( 'success', 'Post has been deleted' );
}
$this->index();
}
}

View File

@ -0,0 +1,50 @@
<?php
/**
* app/plugins/members/controllers/admin/members/records.php
*
* This is the Membership Records admin controller.
*
* @package TP Members
* @version 3.0
* @author Joey Kimsey <Joey@thetempusproject.com>
* @link https://TheTempusProject.com
* @license https://opensource.org/licenses/MIT [MIT LICENSE]
*/
namespace TheTempusProject\Controllers\Admin;
use TheTempusProject\Houdini\Classes\Views;
use TheTempusProject\Houdini\Classes\Navigation;
use TheTempusProject\Houdini\Classes\Components;
use TheTempusProject\Classes\AdminController;
use TheTempusProject\Models\Memberships as MemberModel;
class Records extends AdminController {
public static $memberships;
public function __construct() {
parent::__construct();
self::$title = 'Admin - Memberships';
self::$memberships = new MemberModel;
$view = Navigation::activePageSelect( 'nav.admin', '/admin/member' );
Components::set( 'ADMINNAV', $view );
}
public function index( $data = null ) {
Views::view( 'members.admin.memberships.list', self::$memberships->list() );
}
public function create( $data = null ) {
}
public function edit( $data = null ) {
}
public function view( $data = null ) {
}
public function delete( $data = null ) {
}
public function preview( $data = null ) {
}
}

View File

@ -0,0 +1,188 @@
<?php
/**
* app/controllers/api/users.php
*
* This is the users' api controller.
*
* @version 3.0
* @author Joey Kimsey <Joey@thetempusproject.com>
* @link https://TheTempusProject.com
* @license https://opensource.org/licenses/MIT [MIT LICENSE]
*/
namespace TheTempusProject\Controllers\Api;
use Stripe\StripeClient;
use Stripe\Event;
use TheTempusProject\Models\User;
use TheTempusProject\Controllers\StripeApiController;
use TheTempusProject\Houdini\Classes\Views;
use TheTempusProject\Bedrock\Classes\Config;
use TheTempusProject\Canary\Bin\Canary as Debug;
use TheTempusProject\Models\MembershipCustomers;
use TheTempusProject\Models\Memberships;
class Stripe extends StripeApiController {
public static $stripe;
public static $customers;
public static $memberships;
public function __construct() {
parent::__construct();
$api_key = Config::getValue( 'memberships/stripeSecret' );
self::$stripe = new StripeClient($api_key);
}
public function webhook() {
try {
$payload = @file_get_contents('php://input');
$payload = json_decode( $payload, true );
if ( ! is_array( $payload ) ) {
throw new \Exception("Error Processing Request", 1);
}
$event = null;
$event = Event::constructFrom( $payload );
$eventData = $event->data->object;
// $event->type gives the event type obv
switch ($event->type) {
// case 'invoice.paid':
// Debug::error( 'processing: ' . $event->type );
// "id": "in_1QSDcJGsigymNdIJo0Z1a20K",
// "customer": "cus_RKtOtR7X7CwPRU",
// "charge": "ch_3QSDcJGsigymNdIJ0Smb7Rmx",
// "subscription": "sub_1QSDcJGsigymNdIJWGw7Zrv9",
// break;
// case 'charge.succeeded':
// Debug::error( 'processing: ' . $event->type );
// "id": "ch_3QSDcJGsigymNdIJ0Smb7Rmx",
// "invoice": "in_1QSDcJGsigymNdIJo0Z1a20K",
// "status": "succeeded",
// break;
case 'customer.subscription.updated':
case 'customer.subscription.paused':
case 'customer.subscription.resumed':
case 'customer.subscription.deleted':
Debug::error( 'processing: ' . $event->type );
self::$memberships = new Memberships;
$membership_id = self::$memberships->findBySubscriptionID( $eventData->id );
if ( empty( $membership_id ) ) {
Debug::error( 'membership not found' );
self::$customers = new MembershipCustomers;
$customer = self::$customers->findByCustomerID( $eventData->customer );
if ( empty( $customer ) ) {
Debug::error( 'customer not found' );
Debug::v( $eventData->customer );
break;
}
$result = self::$memberships->create(
$eventData->customer,
$eventData->id,
$eventData->plan->id,
$eventData->current_period_start,
$eventData->current_period_end,
$eventData->status,
$customer->local_user,
'frequency'
);
} else {
$result = self::$memberships->update( $membership_id->ID, $eventData->current_period_start, $eventData->current_period_end, $eventData->status );
}
if ( empty( $result ) ) {
Debug::error( 'membership not updated' );
Debug::v( $result );
}
break;
case 'customer.subscription.created':
Debug::error( 'processing: ' . $event->type );
self::$memberships = new Memberships;
$membership_id = self::$memberships->findBySubscriptionID( $eventData->id );
if ( ! empty( $membership_id ) ) {
Debug::error( 'subscription already created' );
break;
}
self::$customers = new MembershipCustomers;
$customer = self::$customers->findByCustomerID( $eventData->customer );
if ( empty( $customer ) ) {
Debug::error( 'customer not found' );
Debug::v( $eventData->customer );
break;
}
Debug::error( 'processing: ' . $event->type );
$result = self::$memberships->create(
$eventData->customer,
$eventData->id,
$eventData->plan->id,
$eventData->current_period_start,
$eventData->current_period_end,
$eventData->status,
$customer->local_user,
'frequency'
);
Debug::error( 'processing: ' . $event->type );
if ( empty( $result ) ) {
Debug::error( 'membership not made' );
Debug::v( $result );
}
Debug::error( 'done processing: ' . var_export($result,true) );
break;
// case 'invoice.created':
// Debug::error( 'processing: ' . $event->type );
// happens when the payment first starts
// "id": "in_1QSDcJGsigymNdIJo0Z1a20K",
// "customer": "cus_RKtOtR7X7CwPRU",
// "status": "open",
// "total": 888,
// break;
// case 'checkout.session.completed':
// Debug::error( 'processing: ' . $event->type );
// new customer has completed first checkout
// add thier record or update iit
// "invoice": "in_1QSDcJGsigymNdIJo0Z1a20K",
// "mode": "subscription",
// "status": "complete",
// "customer": "cus_RKtOtR7X7CwPRU",
// break;
// case 'payment_intent.succeeded':
// Debug::error( 'processing: ' . $event->type );
// $paymentIntent = $event->data->object;
// break;
// case 'payment_method.attached':
// Debug::error( 'processing: ' . $event->type );
// $paymentMethod = $event->data->object;
// break;
default:
Debug::error( 'Skipped Event:' . $event->type );
break;
}
$responseType = 'success';
$response = true;
} catch(\UnexpectedValueException $e) {
Debug::error( 'UnexpectedValueException' );
Debug::v( $e );
http_response_code(400);
$responseType = 'error';
$response = 'UnexpectedValueException';
} catch(\Exception $e) {
Debug::error( 'Exception' );
Debug::v( $e );
http_response_code(400);
$responseType = 'error';
$response = 'Exception';
}
Views::view( 'api.response', ['response' => json_encode( [ $responseType => $response ], true )]);
}
}

View File

@ -18,19 +18,223 @@ use TheTempusProject\Classes\Controller;
use TheTempusProject\TheTempusProject as App;
use TheTempusProject\Hermes\Functions\Redirect;
use TheTempusProject\Bedrock\Functions\Session;
use TheTempusProject\Models\MembershipCustomers;
use TheTempusProject\Models\MembershipProducts;
use TheTempusProject\Bedrock\Classes\Config;
use TheTempusProject\Bedrock\Functions\Input;
use Stripe\Checkout\Session as StripeSession;
use TheTempusProject\Hermes\Functions\Route as Routes;
use TheTempusProject\Houdini\Classes\Navigation;
use TheTempusProject\Models\Memberships;
use TheTempusProject\Houdini\Classes\Components;
class Member extends Controller {
public static $customers;
public static $products;
public function __construct() {
parent::__construct();
Template::noIndex();
if ( !App::$isMember ) {
Session::flash( 'error', 'You do not have permission to view this page.' );
return Redirect::home();
}
$api_key = Config::getValue( 'memberships/stripeSecret' );
self::$customers = new MembershipCustomers;
self::$products = new MembershipProducts;
}
public function index() {
self::$title = 'Members Area';
if ( !App::$isMember ) {
Session::flash( 'error', 'You do not have permission to view this page.' );
return Redirect::home();
}
Views::view( 'members.members' );
}
}
public function managepayment( $id = null ) {
$api_key = Config::getValue( 'memberships/stripeSecret' );
$stripe = new \Stripe\StripeClient( $api_key );
$customer = self::$customers->findOrCreate( App::$activeUser->ID );
if ( empty( $customer ) ) {
Session::flash( 'error', 'no customer.' );
return Redirect::to( 'member/manage' );
}
$session = $stripe->billingPortal->sessions->create([
'customer' => $customer,
'return_url' => Routes::getAddress() . 'member/manage',
]);
header('Location: ' . $session->url);
exit;
}
public function cancelconfirm( $id = null ) {
$memberships = new Memberships;
$result = $memberships->cancel( $id );
// dv( $result );
if ( ! empty( $result ) ) {
Session::flash( 'success', 'Your Membership has been paused.' );
Redirect::to( 'member/manage' );
} else {
Session::flash( 'error', 'There was an error canceling your membership' );
Redirect::to( 'member/manage' );
}
}
public function pauseconfirm( $id = null ) {
$memberships = new Memberships;
$result = $memberships->cancel( $id );
// dv( $result );
if ( ! empty( $result ) ) {
Session::flash( 'success', 'Your Membership has been paused.' );
Redirect::to( 'member/manage' );
} else {
Session::flash( 'error', 'There was an error canceling your membership' );
Redirect::to( 'member/manage' );
}
}
public function pause( $id = null ) {
self::$title = 'pause Membership';
Components::set( 'pauseid', $id );
Views::view( 'members.pause' );
}
public function resume( $id = null ) {
self::$title = 'resume Membership';
Views::view( 'members.resume' );
}
public function cancel( $id = null ) {
self::$title = 'Cancel Membership';
Components::set( 'cancelid', $id );
Views::view( 'members.cancel' );
}
public function manage( $id = null ) {
self::$title = 'Manage Membership';
$menu = Views::simpleView( 'nav.usercp', App::$userCPlinks );
Navigation::activePageSelect( $menu, null, true, true );
$memberships = new Memberships;
$userMemberships = $memberships->getUserSubs();
Views::view( 'members.manage', $userMemberships );
}
public function upgrade( $id = null ) {
self::$title = 'Upgrade Membership';
Views::view( 'members.upgrade' );
}
public function join( $id = null ) {
self::$title = 'Join {SIITENAME}!';
$product = self::$products->findById( $id );
if ( empty( $product ) ) {
Session::flash( 'success', 'We aren\'t currently accepting new members, please check back soon!' );
return Redirect::home();
}
Views::view( 'members.landing1', $product );
}
public function signup( $id = null ) {
self::$title = 'Sign-up for {SIITENAME}!';
$product = self::$products->findById( $id );
if ( empty( $product ) ) {
Session::flash( 'success', 'We aren\'t currently accepting new members, please check back soon!' );
return Redirect::home();
}
Views::view( 'members.landing2', $product );
}
public function getyearly( $id = null ) {
if ( empty( $id ) ) {
Issues::add( 'error', 'no id' );
return $this->index();
}
$product = self::$products->findById( $id );
if ( empty( $product ) ) {
Issues::add( 'error', 'no product' );
return $this->index();
}
$customer = self::$customers->findOrCreate( App::$activeUser->ID );
if ( empty( $customer ) ) {
Issues::add( 'error', 'no customer' );
return $this->index();
}
self::$title = 'Purchase';
$price = $product->stripe_price_yearly;
$api_key = Config::getValue( 'memberships/stripeSecret' );
$stripe = new \Stripe\StripeClient( $api_key );
$session = $stripe->checkout->sessions->create([
'payment_method_types' => ['card'],
'customer' => $customer,
'line_items' => [[
'price' => $price,
'quantity' => 1,
]],
'mode' => 'subscription',
'success_url' => Routes::getAddress() . 'member/paymentcomplete?session_id={CHECKOUT_SESSION_ID}',
'cancel_url' => Routes::getAddress() . 'member/paymentcanceled',
]);
header('Location: ' . $session->url);
exit;
}
public function getmonthly( $id = null ) {
if ( empty( $id ) ) {
Issues::add( 'error', 'no id' );
return $this->index();
}
$product = self::$products->findById( $id );
if ( empty( $product ) ) {
Issues::add( 'error', 'no product' );
return $this->index();
}
$customer = self::$customers->findOrCreate( App::$activeUser->ID );
if ( empty( $customer ) ) {
Issues::add( 'error', 'no customer' );
return $this->index();
}
self::$title = 'Purchase';
$price = $product->stripe_price_monthly;
$api_key = Config::getValue( 'memberships/stripeSecret' );
$stripe = new \Stripe\StripeClient( $api_key );
$session = $stripe->checkout->sessions->create([
'payment_method_types' => ['card'],
'customer' => $customer,
'line_items' => [[
'price' => $price,
'quantity' => 1,
]],
'mode' => 'subscription',
'success_url' => Routes::getAddress() . 'member/paymentcomplete?session_id={CHECKOUT_SESSION_ID}',
'cancel_url' => Routes::getAddress() . 'member/paymentcanceled',
]);
header('Location: ' . $session->url);
exit;
}
public function paymentcanceled() {
self::$title = '(almost) Members Area';
Views::view( 'members.paymentcanceled' );
}
public function paymentcomplete() {
self::$title = '(almost) Members Area';
Views::view( 'members.paymentcomplete' );
}
}

View File

@ -0,0 +1,33 @@
<?php
/**
* app/classes/admin_controller.php
*
* This is the base admin controller. Every other admin controller should
* extend this class.
*
* @version 3.0
* @author Joey Kimsey <Joey@thetempusproject.com>
* @link https://TheTempusProject.com
* @license https://opensource.org/licenses/MIT [MIT LICENSE]
*/
namespace TheTempusProject\Controllers;
use TheTempusProject\Houdini\Classes\Template;
use TheTempusProject\TheTempusProject as App;
use TheTempusProject\Hermes\Functions\Redirect;
use TheTempusProject\Bedrock\Functions\Session;
use TheTempusProject\Classes\Controller;
class StripeApiController extends Controller {
public function __construct() {
parent::__construct();
// if ( ! App::verifyApiRequest() ) {
// Session::flash( 'error', 'You do not have permission to view this page.' );
// return Redirect::home();
// }
Template::noFollow();
Template::noIndex();
Template::addHeader( 'Content-Type: application/json; charset=utf-8' );
Template::setTemplate( 'api' );
}
}