Compare commits
3 Commits
development
..
rc
| Author | SHA1 | Date | |
|---|---|---|---|
| 629886a7d9 | |||
| 297cc45f7d | |||
| 6d41479838 |
@@ -5,7 +5,7 @@
|
|||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: mokocli.Automation
|
# INGROUP: mokocli.Automation
|
||||||
# VERSION: 01.13.04
|
# VERSION: 01.10.00
|
||||||
# BRIEF: Auto-create feature branch when an issue is opened
|
# BRIEF: Auto-create feature branch when an issue is opened
|
||||||
|
|
||||||
name: "Universal: Issue Branch"
|
name: "Universal: Issue Branch"
|
||||||
|
|||||||
+8
-18
@@ -1,19 +1,7 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
### Fixed
|
## [01.10.00] --- 2026-06-28
|
||||||
- **deleteFromPlatforms()**: Use `CredentialHelper::decrypt()` instead of raw `json_decode` for encrypted credentials (#226)
|
|
||||||
- **deleteFromPlatforms()**: Use Joomla 5/6 `getDispatcher()->dispatch()` instead of deprecated `triggerEvent()` (#228)
|
|
||||||
- **PostsController**: Add ACL checks to `retryFailed()` and `purgePosted()` queue actions (#224)
|
|
||||||
- **QueueProcessor**: Recover stale `posting` entries stuck > 10 minutes back to `queued` (#235)
|
|
||||||
- **onContentChangeState**: Respect `post_on_first_publish_only` setting when state-toggling articles (#238)
|
|
||||||
- **Uninstall SQL**: Add missing `analytics` and `category_rules` table drops (#225)
|
|
||||||
- **Dashboard/Calendar views**: Remove deprecated `Sidebar::render()` calls (#250)
|
|
||||||
- **AnalyticsHelper**: Rewrite AJAX heatmap/best-times to query `#__mokosuitecross_posts` instead of empty `analytics` table (#246)
|
|
||||||
- **Submenu helper**: Remove duplicate `calendar` key in `addSubmenu()` (#248)
|
|
||||||
- **CHANGELOG**: Remove 3 duplicate version headers (#240)
|
|
||||||
|
|
||||||
## [01.12.00] --- 2026-06-28
|
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
- **Visual post calendar**: Monthly calendar grid view showing scheduled, queued, and posted cross-posts with status badges (#160)
|
- **Visual post calendar**: Monthly calendar grid view showing scheduled, queued, and posted cross-posts with status badges (#160)
|
||||||
@@ -22,9 +10,8 @@
|
|||||||
- **Analytics service filter**: Filter heatmap and stats by service type with configurable date range
|
- **Analytics service filter**: Filter heatmap and stats by service type with configurable date range
|
||||||
- **Analytics service breakdown**: Per-service success rate, failure count, and average posts per day
|
- **Analytics service breakdown**: Per-service success rate, failure count, and average posts per day
|
||||||
- **Analytics AJAX endpoint**: JSON heatmap data for dynamic filtering without page reload
|
- **Analytics AJAX endpoint**: JSON heatmap data for dynamic filtering without page reload
|
||||||
- **Social image generator**: Generate branded 1200x630 OG images with article title overlay using PHP GD (#157)
|
- **Social image generator**: Generate Open Graph images with article title overlay using PHP GD library (#157)
|
||||||
- **Social image config**: Background color, text color, font size, and site name branding options (#157)
|
- **Social image config**: Background color, text color, overlay style, and site name override in component options (#157)
|
||||||
- **Generate Social Image button**: One-click image generation in the Share Content panel (#157)
|
|
||||||
- **AI caption generation**: Generate platform-optimized cross-post captions from article content using Claude or OpenAI (#161)
|
- **AI caption generation**: Generate platform-optimized cross-post captions from article content using Claude or OpenAI (#161)
|
||||||
- **AI provider config**: New "AI Caption Generation" fieldset in component options with provider, API key, model, and tone settings
|
- **AI provider config**: New "AI Caption Generation" fieldset in component options with provider, API key, model, and tone settings
|
||||||
- **AI Generate button**: One-click AI generation button in the Share Content panel that fills all caption fields
|
- **AI Generate button**: One-click AI generation button in the Share Content panel that fills all caption fields
|
||||||
@@ -57,9 +44,10 @@
|
|||||||
- **PHPUnit test suite**: Unit tests for models, helpers, and service plugins (#132)
|
- **PHPUnit test suite**: Unit tests for models, helpers, and service plugins (#132)
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
- **Content plugin**: Remove Joomla 5 typed event hints -- Joomla 6 dispatches `Model\AfterSaveEvent` instead of `Content\AfterSaveEvent`, causing fatal TypeError on article save
|
|
||||||
- **PreviewController**: Add ACL check and parameterized query to prevent unauthorized article access (IDOR)
|
- **PreviewController**: Add ACL check and parameterized query to prevent unauthorized article access (IDOR)
|
||||||
- Webservices plugin Joomla 6 compatibility -- `onBeforeApiRoute` receives `BeforeApiRouteEvent` object, extract router via `$event->getRouter()`
|
- Webservices plugin Joomla 6 compatibility — `onBeforeApiRoute` receives `BeforeApiRouteEvent` object, extract router via `$event->getRouter()`
|
||||||
|
|
||||||
|
## [01.07.00] --- 2026-06-23
|
||||||
|
|
||||||
## [01.07.00] --- 2026-06-23
|
## [01.07.00] --- 2026-06-23
|
||||||
|
|
||||||
@@ -74,6 +62,8 @@
|
|||||||
|
|
||||||
## [01.05.00] --- 2026-06-23
|
## [01.05.00] --- 2026-06-23
|
||||||
|
|
||||||
|
## [01.05.00] --- 2026-06-23
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
- **Instagram plugin**: Cross-post to Instagram via Meta Content Publishing API (2-step container flow)
|
- **Instagram plugin**: Cross-post to Instagram via Meta Content Publishing API (2-step container flow)
|
||||||
- **YouTube plugin**: Cross-post to YouTube via Data API v3 channel bulletins
|
- **YouTube plugin**: Cross-post to YouTube via Data API v3 channel bulletins
|
||||||
|
|||||||
+1
-1
@@ -14,7 +14,7 @@
|
|||||||
DEFGROUP: Template-Joomla
|
DEFGROUP: Template-Joomla
|
||||||
INGROUP: Template-Joomla.Documentation
|
INGROUP: Template-Joomla.Documentation
|
||||||
REPO: https://github.com/mokoconsulting-tech/Template-Joomla/
|
REPO: https://github.com/mokoconsulting-tech/Template-Joomla/
|
||||||
VERSION: 01.13.04
|
VERSION: 01.10.00
|
||||||
PATH: ./CODE_OF_CONDUCT.md
|
PATH: ./CODE_OF_CONDUCT.md
|
||||||
BRIEF: Community expectations and enforcement guidelines
|
BRIEF: Community expectations and enforcement guidelines
|
||||||
NOTE: Adapted with attribution from the Contributor Covenant v2.1
|
NOTE: Adapted with attribution from the Contributor Covenant v2.1
|
||||||
|
|||||||
+1
-1
@@ -19,7 +19,7 @@
|
|||||||
DEFGROUP: mokoconsulting-tech.Template-Joomla
|
DEFGROUP: mokoconsulting-tech.Template-Joomla
|
||||||
INGROUP: MokoStandards.Governance
|
INGROUP: MokoStandards.Governance
|
||||||
REPO: https://github.com/mokoconsulting-tech/Template-Joomla
|
REPO: https://github.com/mokoconsulting-tech/Template-Joomla
|
||||||
VERSION: 01.13.04
|
VERSION: 01.10.00
|
||||||
PATH: /GOVERNANCE.md
|
PATH: /GOVERNANCE.md
|
||||||
BRIEF: Project governance rules, roles, and decision process for Template-Joomla
|
BRIEF: Project governance rules, roles, and decision process for Template-Joomla
|
||||||
-->
|
-->
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
# MokoSuiteCross
|
# MokoSuiteCross
|
||||||
|
|
||||||
<!-- VERSION: 01.13.04 -->
|
<!-- VERSION: 01.10.00 -->
|
||||||
|
|
||||||
Cross-posting Joomla content to social media, email marketing, and chat platforms for Joomla 6.
|
Cross-posting Joomla content to social media, email marketing, and chat platforms for Joomla 5/6.
|
||||||
|
|
||||||
## Overview
|
## Overview
|
||||||
|
|
||||||
|
|||||||
+1
-1
@@ -23,7 +23,7 @@ DEFGROUP: Template-Joomla
|
|||||||
INGROUP: Template-Joomla.Documentation
|
INGROUP: Template-Joomla.Documentation
|
||||||
REPO: https://git.mokoconsulting.tech/MokoConsulting/Template-Joomla
|
REPO: https://git.mokoconsulting.tech/MokoConsulting/Template-Joomla
|
||||||
PATH: /SECURITY.md
|
PATH: /SECURITY.md
|
||||||
VERSION: 01.13.04
|
VERSION: 01.10.00
|
||||||
BRIEF: Security vulnerability reporting and handling policy
|
BRIEF: Security vulnerability reporting and handling policy
|
||||||
-->
|
-->
|
||||||
|
|
||||||
|
|||||||
@@ -266,7 +266,7 @@
|
|||||||
</field>
|
</field>
|
||||||
</fieldset>
|
</fieldset>
|
||||||
|
|
||||||
<fieldset name="social_image" label="COM_MOKOSUITECROSS_CONFIG_SOCIAL_IMAGE">
|
<fieldset name="social_image" label="COM_MOKOSUITECROSS_CONFIG_SOCIAL_IMAGE">
|
||||||
<field
|
<field
|
||||||
name="social_image_enabled"
|
name="social_image_enabled"
|
||||||
type="radio"
|
type="radio"
|
||||||
|
|||||||
@@ -572,20 +572,38 @@ COM_MOKOSUITECROSS_AI_GENERATED="AI captions generated successfully."
|
|||||||
COM_MOKOSUITECROSS_AI_ERROR="AI generation failed: %s"
|
COM_MOKOSUITECROSS_AI_ERROR="AI generation failed: %s"
|
||||||
COM_MOKOSUITECROSS_AI_NOT_CONFIGURED="AI is not configured. Go to Options to set up a provider and API key."
|
COM_MOKOSUITECROSS_AI_NOT_CONFIGURED="AI is not configured. Go to Options to set up a provider and API key."
|
||||||
|
|
||||||
|
; Analytics
|
||||||
|
COM_MOKOSUITECROSS_SUBMENU_ANALYTICS="Analytics"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_PERIOD="Time Period"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_SERVICE_FILTER="Service"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_ALL_SERVICES="All Services"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_BEST_TIMES="Best Times to Post"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_HEATMAP="Engagement Heatmap"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_HOURLY="Hourly Distribution"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_DAILY="Day of Week Distribution"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_NO_DATA="Not enough posting data to generate recommendations. Post at least 3 times per time slot over the selected period."
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_POSTS_SUCCESS="%d of %d successful"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_DAY_SUN="Sun"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_DAY_MON="Mon"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_DAY_TUE="Tue"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_DAY_WED="Wed"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_DAY_THU="Thu"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_DAY_FRI="Fri"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_DAY_SAT="Sat"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_LEGEND_HIGH="High success rate"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_LEGEND_MEDIUM="Medium success rate"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_LEGEND_LOW="Low success rate"
|
||||||
|
COM_MOKOSUITECROSS_ANALYTICS_LEGEND_NONE="No data"
|
||||||
|
COM_MOKOSUITECROSS_PERIOD_180_DAYS="Last 180 days"
|
||||||
|
COM_MOKOSUITECROSS_PERIOD_365_DAYS="Last 365 days"
|
||||||
|
|
||||||
; Category Rules
|
; Category Rules
|
||||||
COM_MOKOSUITECROSS_CONFIG_CATEGORY_RULES="Category Rules"
|
COM_MOKOSUITECROSS_CONFIG_CATEGORY_RULES="Category Rules"
|
||||||
COM_MOKOSUITECROSS_CONFIG_CATEGORY_RULES_NOTE="Category Routing"
|
COM_MOKOSUITECROSS_CONFIG_CATEGORY_RULES_NOTE="Category Routing"
|
||||||
COM_MOKOSUITECROSS_CONFIG_CATEGORY_RULES_NOTE_DESC="Category routing rules let you map Joomla categories to specific cross-post services. When rules exist for a category, only those services receive posts. When no rules exist, all services are used (default behaviour). Rules are managed in the database table #__mokosuitecross_category_rules. A full admin UI will be added in a future release."
|
COM_MOKOSUITECROSS_CONFIG_CATEGORY_RULES_NOTE_DESC="Category routing rules let you map Joomla categories to specific cross-post services. When rules exist for a category, only those services receive posts. When no rules exist, all services are used (default behaviour). Rules are managed in the database table #__mokosuitecross_category_rules. A full admin UI will be added in a future release."
|
||||||
|
|
||||||
; Post Calendar
|
; Calendar View
|
||||||
COM_MOKOSUITECROSS_CALENDAR="Post Calendar"
|
COM_MOKOSUITECROSS_CALENDAR_PREV_MONTH="Previous"
|
||||||
COM_MOKOSUITECROSS_CALENDAR_DESC="Visual calendar of scheduled and posted content"
|
COM_MOKOSUITECROSS_CALENDAR_NEXT_MONTH="Next"
|
||||||
COM_MOKOSUITECROSS_SUBMENU_CALENDAR="Calendar"
|
|
||||||
COM_MOKOSUITECROSS_CALENDAR_TODAY="Today"
|
COM_MOKOSUITECROSS_CALENDAR_TODAY="Today"
|
||||||
COM_MOKOSUITECROSS_CALENDAR_MONTH="Month"
|
COM_MOKOSUITECROSS_SUBMENU_CALENDAR="Post Calendar"
|
||||||
COM_MOKOSUITECROSS_CALENDAR_WEEK="Week"
|
|
||||||
COM_MOKOSUITECROSS_CALENDAR_LIST="List"
|
|
||||||
COM_MOKOSUITECROSS_CALENDAR_RESCHEDULE_SUCCESS="Post rescheduled successfully"
|
|
||||||
COM_MOKOSUITECROSS_CALENDAR_RESCHEDULE_ERROR="Failed to reschedule post"
|
|
||||||
COM_MOKOSUITECROSS_CALENDAR_CANNOT_RESCHEDULE="Only scheduled or queued posts can be rescheduled"
|
|
||||||
COM_MOKOSUITECROSS_CALENDAR_LOAD_ERROR="Failed to load calendar events"
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="component" method="upgrade">
|
<extension type="component" method="upgrade">
|
||||||
<name>com_mokosuitecross</name>
|
<name>com_mokosuitecross</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -96,27 +96,6 @@ INSERT INTO `#__mokosuitecross_templates` (`service_type`, `title`, `template_bo
|
|||||||
('instagram', 'Instagram Default', '{social}\n\n{hashtags}', 1, 21, NOW()),
|
('instagram', 'Instagram Default', '{social}\n\n{hashtags}', 1, 21, NOW()),
|
||||||
('youtube', 'YouTube Default', '{social}\n\n{url}', 1, 22, NOW());
|
('youtube', 'YouTube Default', '{social}\n\n{url}', 1, 22, NOW());
|
||||||
|
|
||||||
|
|
||||||
CREATE TABLE IF NOT EXISTS `#__mokosuitecross_analytics` (
|
|
||||||
`id` int unsigned NOT NULL AUTO_INCREMENT,
|
|
||||||
`post_id` int unsigned NOT NULL,
|
|
||||||
`service_id` int unsigned NOT NULL,
|
|
||||||
`service_type` varchar(50) NOT NULL DEFAULT '',
|
|
||||||
`posted_at` datetime DEFAULT NULL,
|
|
||||||
`day_of_week` tinyint unsigned NOT NULL DEFAULT 0,
|
|
||||||
`hour_of_day` tinyint unsigned NOT NULL DEFAULT 0,
|
|
||||||
`impressions` int unsigned NOT NULL DEFAULT 0,
|
|
||||||
`engagements` int unsigned NOT NULL DEFAULT 0,
|
|
||||||
`clicks` int unsigned NOT NULL DEFAULT 0,
|
|
||||||
`shares` int unsigned NOT NULL DEFAULT 0,
|
|
||||||
`engagement_rate` decimal(5,2) NOT NULL DEFAULT 0.00,
|
|
||||||
`created` datetime NOT NULL,
|
|
||||||
PRIMARY KEY (`id`),
|
|
||||||
KEY `idx_service_type` (`service_type`),
|
|
||||||
KEY `idx_day_hour` (`day_of_week`, `hour_of_day`),
|
|
||||||
KEY `idx_post` (`post_id`)
|
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
|
|
||||||
|
|
||||||
CREATE TABLE IF NOT EXISTS `#__mokosuitecross_category_rules` (
|
CREATE TABLE IF NOT EXISTS `#__mokosuitecross_category_rules` (
|
||||||
`id` int(10) unsigned NOT NULL AUTO_INCREMENT,
|
`id` int(10) unsigned NOT NULL AUTO_INCREMENT,
|
||||||
`category_id` int(10) unsigned NOT NULL,
|
`category_id` int(10) unsigned NOT NULL,
|
||||||
|
|||||||
@@ -1,7 +1,5 @@
|
|||||||
-- MokoSuiteCross -- Uninstall
|
-- MokoSuiteCross — Uninstall
|
||||||
DROP TABLE IF EXISTS `#__mokosuitecross_logs`;
|
DROP TABLE IF EXISTS `#__mokosuitecross_logs`;
|
||||||
DROP TABLE IF EXISTS `#__mokosuitecross_analytics`;
|
|
||||||
DROP TABLE IF EXISTS `#__mokosuitecross_category_rules`;
|
|
||||||
DROP TABLE IF EXISTS `#__mokosuitecross_posts`;
|
DROP TABLE IF EXISTS `#__mokosuitecross_posts`;
|
||||||
DROP TABLE IF EXISTS `#__mokosuitecross_templates`;
|
DROP TABLE IF EXISTS `#__mokosuitecross_templates`;
|
||||||
DROP TABLE IF EXISTS `#__mokosuitecross_services`;
|
DROP TABLE IF EXISTS `#__mokosuitecross_services`;
|
||||||
|
|||||||
@@ -1,23 +1 @@
|
|||||||
-- MokoSuiteCross 01.08.54 -- Best time to post analytics
|
/* 01.08.54 — no schema changes */
|
||||||
-- Copyright (C) 2026 Moko Consulting. All rights reserved.
|
|
||||||
-- SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
CREATE TABLE IF NOT EXISTS `#__mokosuitecross_analytics` (
|
|
||||||
`id` int unsigned NOT NULL AUTO_INCREMENT,
|
|
||||||
`post_id` int unsigned NOT NULL,
|
|
||||||
`service_id` int unsigned NOT NULL,
|
|
||||||
`service_type` varchar(50) NOT NULL DEFAULT '',
|
|
||||||
`posted_at` datetime DEFAULT NULL,
|
|
||||||
`day_of_week` tinyint unsigned NOT NULL DEFAULT 0,
|
|
||||||
`hour_of_day` tinyint unsigned NOT NULL DEFAULT 0,
|
|
||||||
`impressions` int unsigned NOT NULL DEFAULT 0,
|
|
||||||
`engagements` int unsigned NOT NULL DEFAULT 0,
|
|
||||||
`clicks` int unsigned NOT NULL DEFAULT 0,
|
|
||||||
`shares` int unsigned NOT NULL DEFAULT 0,
|
|
||||||
`engagement_rate` decimal(5,2) NOT NULL DEFAULT 0.00,
|
|
||||||
`created` datetime NOT NULL,
|
|
||||||
PRIMARY KEY (`id`),
|
|
||||||
KEY `idx_service_type` (`service_type`),
|
|
||||||
KEY `idx_day_hour` (`day_of_week`, `hour_of_day`),
|
|
||||||
KEY `idx_post` (`post_id`)
|
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
|
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
/* 01.08.59 — no schema changes */
|
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
/* 01.08.62 — no schema changes */
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
/* 01.09.00 — no schema changes */
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
/* 01.10.00 — no schema changes */
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 01.11.00 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 01.11.01 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 01.11.02 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 01.11.03 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 01.12.00 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 01.12.01 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 01.12.03 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 01.12.04 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 01.12.05 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 01.12.06 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 01.13.04 — no schema changes */
|
|
||||||
@@ -14,84 +14,11 @@ namespace Joomla\Component\MokoSuiteCross\Administrator\Controller;
|
|||||||
defined('_JEXEC') or die;
|
defined('_JEXEC') or die;
|
||||||
|
|
||||||
use Joomla\CMS\MVC\Controller\BaseController;
|
use Joomla\CMS\MVC\Controller\BaseController;
|
||||||
use Joomla\CMS\Session\Session;
|
|
||||||
use Joomla\Component\MokoSuiteCross\Administrator\Helper\AnalyticsHelper;
|
|
||||||
|
|
||||||
class AnalyticsController extends BaseController
|
class AnalyticsController extends BaseController
|
||||||
{
|
{
|
||||||
/**
|
public function display($cachable = false, $urlparams = []): static
|
||||||
* Return heatmap grid data as JSON.
|
|
||||||
*
|
|
||||||
* Query params: service_type (string), days (int, default 90)
|
|
||||||
*/
|
|
||||||
public function heatmap(): void
|
|
||||||
{
|
{
|
||||||
if (!Session::checkToken('get')) {
|
return parent::display($cachable, $urlparams);
|
||||||
echo json_encode(['success' => false, 'error' => 'Invalid token']);
|
|
||||||
$this->app->close();
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$user = $this->app->getIdentity();
|
|
||||||
|
|
||||||
if (!$user->authorise('core.manage', 'com_mokosuitecross')) {
|
|
||||||
echo json_encode(['success' => false, 'error' => 'Permission denied']);
|
|
||||||
$this->app->close();
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$serviceType = $this->input->getCmd('service_type', '');
|
|
||||||
$days = $this->input->getInt('days', 90);
|
|
||||||
|
|
||||||
$grid = AnalyticsHelper::getHeatmapData($serviceType, $days);
|
|
||||||
$bestTimes = AnalyticsHelper::getBestTimes($serviceType, 3);
|
|
||||||
|
|
||||||
$this->app->setHeader('Content-Type', 'application/json; charset=utf-8');
|
|
||||||
echo json_encode([
|
|
||||||
'success' => true,
|
|
||||||
'grid' => $grid,
|
|
||||||
'best_times' => $bestTimes,
|
|
||||||
]);
|
|
||||||
$this->app->close();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the top posting times as JSON.
|
|
||||||
*
|
|
||||||
* Query params: service_type (string), limit (int, default 5)
|
|
||||||
*/
|
|
||||||
public function besttimes(): void
|
|
||||||
{
|
|
||||||
if (!Session::checkToken('get')) {
|
|
||||||
echo json_encode(['success' => false, 'error' => 'Invalid token']);
|
|
||||||
$this->app->close();
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$user = $this->app->getIdentity();
|
|
||||||
|
|
||||||
if (!$user->authorise('core.manage', 'com_mokosuitecross')) {
|
|
||||||
echo json_encode(['success' => false, 'error' => 'Permission denied']);
|
|
||||||
$this->app->close();
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$serviceType = $this->input->getCmd('service_type', '');
|
|
||||||
$limit = $this->input->getInt('limit', 5);
|
|
||||||
|
|
||||||
$bestTimes = AnalyticsHelper::getBestTimes($serviceType, $limit);
|
|
||||||
$serviceBreakdown = AnalyticsHelper::getServiceBreakdown();
|
|
||||||
|
|
||||||
$this->app->setHeader('Content-Type', 'application/json; charset=utf-8');
|
|
||||||
echo json_encode([
|
|
||||||
'success' => true,
|
|
||||||
'best_times' => $bestTimes,
|
|
||||||
'service_breakdown' => $serviceBreakdown,
|
|
||||||
]);
|
|
||||||
$this->app->close();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,256 +13,12 @@ namespace Joomla\Component\MokoSuiteCross\Administrator\Controller;
|
|||||||
|
|
||||||
defined('_JEXEC') or die;
|
defined('_JEXEC') or die;
|
||||||
|
|
||||||
use Joomla\CMS\Factory;
|
|
||||||
use Joomla\CMS\Language\Text;
|
|
||||||
use Joomla\CMS\MVC\Controller\BaseController;
|
use Joomla\CMS\MVC\Controller\BaseController;
|
||||||
use Joomla\CMS\Session\Session;
|
|
||||||
use Joomla\Component\MokoSuiteCross\Administrator\Table\PostTable;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Calendar controller -- provides AJAX endpoints for the visual post calendar.
|
|
||||||
*
|
|
||||||
* Endpoints:
|
|
||||||
* task=calendar.events -- GET JSON feed for FullCalendar (filtered by start/end)
|
|
||||||
* task=calendar.reschedule -- POST reschedule a post to a new date/time
|
|
||||||
*/
|
|
||||||
class CalendarController extends BaseController
|
class CalendarController extends BaseController
|
||||||
{
|
{
|
||||||
/**
|
public function display($cachable = false, $urlparams = []): static
|
||||||
* Return posts as FullCalendar-compatible JSON events.
|
|
||||||
*
|
|
||||||
* Query params: start, end (ISO 8601 date range from FullCalendar).
|
|
||||||
*
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public function events(): void
|
|
||||||
{
|
{
|
||||||
$app = $this->app;
|
return parent::display($cachable, $urlparams);
|
||||||
$db = Factory::getDbo();
|
|
||||||
|
|
||||||
// ACL check
|
|
||||||
if (!$app->getIdentity()->authorise('core.manage', 'com_mokosuitecross')) {
|
|
||||||
$this->sendJsonResponse(['error' => 'Forbidden'], 403);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// FullCalendar sends start/end as ISO date strings
|
|
||||||
$start = $this->input->getString('start', '');
|
|
||||||
$end = $this->input->getString('end', '');
|
|
||||||
|
|
||||||
$query = $db->getQuery(true)
|
|
||||||
->select([
|
|
||||||
'p.' . $db->quoteName('id'),
|
|
||||||
'p.' . $db->quoteName('article_id'),
|
|
||||||
'p.' . $db->quoteName('service_id'),
|
|
||||||
'p.' . $db->quoteName('status'),
|
|
||||||
'p.' . $db->quoteName('scheduled_at'),
|
|
||||||
'p.' . $db->quoteName('posted_at'),
|
|
||||||
'p.' . $db->quoteName('created'),
|
|
||||||
'p.' . $db->quoteName('message'),
|
|
||||||
'a.' . $db->quoteName('title', 'article_title'),
|
|
||||||
's.' . $db->quoteName('title', 'service_title'),
|
|
||||||
's.' . $db->quoteName('service_type'),
|
|
||||||
])
|
|
||||||
->from($db->quoteName('#__mokosuitecross_posts', 'p'))
|
|
||||||
->leftJoin(
|
|
||||||
$db->quoteName('#__content', 'a')
|
|
||||||
. ' ON ' . $db->quoteName('a.id') . ' = ' . $db->quoteName('p.article_id')
|
|
||||||
)
|
|
||||||
->leftJoin(
|
|
||||||
$db->quoteName('#__mokosuitecross_services', 's')
|
|
||||||
. ' ON ' . $db->quoteName('s.id') . ' = ' . $db->quoteName('p.service_id')
|
|
||||||
)
|
|
||||||
->order($db->quoteName('p.created') . ' DESC');
|
|
||||||
|
|
||||||
// Filter by date range when provided
|
|
||||||
if ($start !== '') {
|
|
||||||
$dateExpr = 'COALESCE(p.scheduled_at, p.posted_at, p.created)';
|
|
||||||
$query->where($dateExpr . ' >= ' . $db->quote($start));
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($end !== '') {
|
|
||||||
$dateExpr = 'COALESCE(p.scheduled_at, p.posted_at, p.created)';
|
|
||||||
$query->where($dateExpr . ' <= ' . $db->quote($end));
|
|
||||||
}
|
|
||||||
|
|
||||||
$db->setQuery($query);
|
|
||||||
$rows = $db->loadObjectList() ?: [];
|
|
||||||
|
|
||||||
// Map status to colour
|
|
||||||
$statusColors = [
|
|
||||||
'posted' => '#28a745',
|
|
||||||
'scheduled' => '#007bff',
|
|
||||||
'queued' => '#ffc107',
|
|
||||||
'failed' => '#dc3545',
|
|
||||||
'posting' => '#17a2b8',
|
|
||||||
];
|
|
||||||
|
|
||||||
$events = [];
|
|
||||||
|
|
||||||
foreach ($rows as $row) {
|
|
||||||
// Pick the best date for the calendar event
|
|
||||||
$eventDate = $row->scheduled_at ?: ($row->posted_at ?: $row->created);
|
|
||||||
|
|
||||||
// Skip rows with no usable date
|
|
||||||
if (empty($eventDate) || $eventDate === '0000-00-00 00:00:00') {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$title = ($row->article_title ?: 'Post #' . $row->id);
|
|
||||||
|
|
||||||
if ($row->service_title) {
|
|
||||||
$title .= ' - ' . $row->service_title;
|
|
||||||
}
|
|
||||||
|
|
||||||
$events[] = [
|
|
||||||
'id' => (int) $row->id,
|
|
||||||
'title' => $title,
|
|
||||||
'start' => $eventDate,
|
|
||||||
'color' => $statusColors[$row->status] ?? '#6c757d',
|
|
||||||
'url' => 'index.php?option=com_mokosuitecross&task=post.edit&id=' . (int) $row->id,
|
|
||||||
'extendedProps' => [
|
|
||||||
'status' => $row->status,
|
|
||||||
'service_type' => $row->service_type ?? '',
|
|
||||||
'article_id' => (int) $row->article_id,
|
|
||||||
'service_id' => (int) $row->service_id,
|
|
||||||
'message' => mb_substr($row->message ?? '', 0, 200),
|
|
||||||
],
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->sendJsonResponse($events, 200);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Reschedule a post to a new date/time via drag-drop.
|
|
||||||
*
|
|
||||||
* POST params: post_id (int), new_date (ISO 8601 datetime string).
|
|
||||||
*
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public function reschedule(): void
|
|
||||||
{
|
|
||||||
$app = $this->app;
|
|
||||||
|
|
||||||
// CSRF check
|
|
||||||
if (!Session::checkToken('post')) {
|
|
||||||
$this->sendJsonResponse(['error' => Text::_('JINVALID_TOKEN')], 403);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// ACL check
|
|
||||||
if (!$app->getIdentity()->authorise('core.edit', 'com_mokosuitecross')) {
|
|
||||||
$this->sendJsonResponse(['error' => 'Forbidden'], 403);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$postId = $this->input->getInt('post_id', 0);
|
|
||||||
$newDate = $this->input->getString('new_date', '');
|
|
||||||
|
|
||||||
if ($postId < 1 || $newDate === '') {
|
|
||||||
$this->sendJsonResponse(
|
|
||||||
['error' => Text::_('COM_MOKOSUITECROSS_CALENDAR_RESCHEDULE_ERROR')],
|
|
||||||
400
|
|
||||||
);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Validate the date format
|
|
||||||
try {
|
|
||||||
$dateObj = Factory::getDate($newDate);
|
|
||||||
} catch (\Exception $e) {
|
|
||||||
$this->sendJsonResponse(
|
|
||||||
['error' => Text::_('COM_MOKOSUITECROSS_CALENDAR_RESCHEDULE_ERROR')],
|
|
||||||
400
|
|
||||||
);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Load the post using Table bind/check/store pattern
|
|
||||||
$db = Factory::getDbo();
|
|
||||||
$table = new PostTable($db);
|
|
||||||
|
|
||||||
if (!$table->load($postId)) {
|
|
||||||
$this->sendJsonResponse(
|
|
||||||
['error' => Text::_('COM_MOKOSUITECROSS_CALENDAR_RESCHEDULE_ERROR')],
|
|
||||||
404
|
|
||||||
);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Only allow rescheduling of scheduled or queued posts
|
|
||||||
$allowedStatuses = ['scheduled', 'queued'];
|
|
||||||
|
|
||||||
if (!in_array($table->status, $allowedStatuses, true)) {
|
|
||||||
$this->sendJsonResponse(
|
|
||||||
['error' => Text::_('COM_MOKOSUITECROSS_CALENDAR_RESCHEDULE_ERROR')],
|
|
||||||
400
|
|
||||||
);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update the post
|
|
||||||
$data = [
|
|
||||||
'scheduled_at' => $dateObj->toSql(),
|
|
||||||
'status' => 'scheduled',
|
|
||||||
'modified' => Factory::getDate()->toSql(),
|
|
||||||
];
|
|
||||||
|
|
||||||
if (!$table->bind($data) || !$table->check() || !$table->store()) {
|
|
||||||
$this->sendJsonResponse(
|
|
||||||
['error' => Text::_('COM_MOKOSUITECROSS_CALENDAR_RESCHEDULE_ERROR')],
|
|
||||||
500
|
|
||||||
);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Log the reschedule
|
|
||||||
$log = (object) [
|
|
||||||
'post_id' => $postId,
|
|
||||||
'service_id' => (int) $table->service_id,
|
|
||||||
'level' => 'info',
|
|
||||||
'message' => sprintf('Post rescheduled to %s via calendar drag-drop', $dateObj->toSql()),
|
|
||||||
'context' => '{}',
|
|
||||||
'created' => Factory::getDate()->toSql(),
|
|
||||||
];
|
|
||||||
|
|
||||||
$db->insertObject('#__mokosuitecross_logs', $log);
|
|
||||||
|
|
||||||
$this->sendJsonResponse(
|
|
||||||
[
|
|
||||||
'success' => true,
|
|
||||||
'message' => Text::_('COM_MOKOSUITECROSS_CALENDAR_RESCHEDULE_SUCCESS'),
|
|
||||||
],
|
|
||||||
200
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Send a JSON response and close the application.
|
|
||||||
*
|
|
||||||
* @param array $data Response data
|
|
||||||
* @param int $httpCode HTTP status code
|
|
||||||
*
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
private function sendJsonResponse(array $data, int $httpCode): void
|
|
||||||
{
|
|
||||||
$app = $this->app;
|
|
||||||
|
|
||||||
$app->setHeader('Content-Type', 'application/json; charset=utf-8');
|
|
||||||
$app->setHeader('Status', (string) $httpCode);
|
|
||||||
|
|
||||||
echo json_encode($data, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES);
|
|
||||||
|
|
||||||
$app->close();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -130,10 +130,6 @@ class PostsController extends AdminController
|
|||||||
{
|
{
|
||||||
$this->checkToken();
|
$this->checkToken();
|
||||||
|
|
||||||
if (!$this->app->getIdentity()->authorise('mokosuitecross.queue.manage', 'com_mokosuitecross')) {
|
|
||||||
throw new \RuntimeException(Text::_('JLIB_APPLICATION_ERROR_ACCESS_FORBIDDEN'), 403);
|
|
||||||
}
|
|
||||||
|
|
||||||
$db = Factory::getDbo();
|
$db = Factory::getDbo();
|
||||||
|
|
||||||
$query = $db->getQuery(true)
|
$query = $db->getQuery(true)
|
||||||
@@ -242,10 +238,6 @@ class PostsController extends AdminController
|
|||||||
{
|
{
|
||||||
$this->checkToken();
|
$this->checkToken();
|
||||||
|
|
||||||
if (!$this->app->getIdentity()->authorise('mokosuitecross.queue.manage', 'com_mokosuitecross')) {
|
|
||||||
throw new \RuntimeException(Text::_('JLIB_APPLICATION_ERROR_ACCESS_FORBIDDEN'), 403);
|
|
||||||
}
|
|
||||||
|
|
||||||
$db = Factory::getDbo();
|
$db = Factory::getDbo();
|
||||||
|
|
||||||
$query = $db->getQuery(true)
|
$query = $db->getQuery(true)
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ use Joomla\CMS\Component\ComponentHelper;
|
|||||||
use Joomla\CMS\Factory;
|
use Joomla\CMS\Factory;
|
||||||
use Joomla\CMS\MVC\Controller\BaseController;
|
use Joomla\CMS\MVC\Controller\BaseController;
|
||||||
use Joomla\CMS\Session\Session;
|
use Joomla\CMS\Session\Session;
|
||||||
|
use Joomla\CMS\Uri\Uri;
|
||||||
use Joomla\Component\MokoSuiteCross\Administrator\Helper\SocialImageHelper;
|
use Joomla\Component\MokoSuiteCross\Administrator\Helper\SocialImageHelper;
|
||||||
|
|
||||||
class SocialImageController extends BaseController
|
class SocialImageController extends BaseController
|
||||||
@@ -32,7 +33,7 @@ class SocialImageController extends BaseController
|
|||||||
|
|
||||||
$user = $this->app->getIdentity();
|
$user = $this->app->getIdentity();
|
||||||
|
|
||||||
if (!$user->authorise('core.edit', 'com_mokosuitecross')) {
|
if (!$user->authorise('core.manage', 'com_mokosuitecross')) {
|
||||||
echo json_encode(['success' => false, 'error' => 'Permission denied']);
|
echo json_encode(['success' => false, 'error' => 'Permission denied']);
|
||||||
$this->app->close();
|
$this->app->close();
|
||||||
|
|
||||||
@@ -48,40 +49,47 @@ class SocialImageController extends BaseController
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
$params = ComponentHelper::getParams('com_mokosuitecross');
|
|
||||||
|
|
||||||
if (!(int) $params->get('social_image_enabled', 0)) {
|
|
||||||
echo json_encode(['success' => false, 'error' => 'Social image generator is not enabled']);
|
|
||||||
$this->app->close();
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$db = Factory::getDbo();
|
$db = Factory::getDbo();
|
||||||
$query = $db->getQuery(true)
|
$query = $db->getQuery(true)
|
||||||
->select($db->quoteName('title'))
|
->select($db->quoteName(['id', 'title', 'images']))
|
||||||
->from($db->quoteName('#__content'))
|
->from($db->quoteName('#__content'))
|
||||||
->where($db->quoteName('id') . ' = ' . $articleId);
|
->where($db->quoteName('id') . ' = ' . $articleId);
|
||||||
$db->setQuery($query);
|
$db->setQuery($query);
|
||||||
$title = $db->loadResult();
|
$article = $db->loadObject();
|
||||||
|
|
||||||
if (!$title) {
|
if (!$article) {
|
||||||
echo json_encode(['success' => false, 'error' => 'Article not found']);
|
echo json_encode(['success' => false, 'error' => 'Article not found']);
|
||||||
$this->app->close();
|
$this->app->close();
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
$siteName = $this->app->get('sitename', '');
|
$params = ComponentHelper::getParams('com_mokosuitecross');
|
||||||
|
$siteName = $params->get('social_image_site_name', '') ?: Factory::getApplication()->get('sitename', '');
|
||||||
|
|
||||||
$config = [
|
$options = [
|
||||||
'bg_color' => $params->get('social_image_bg_color', '#1a1a2e'),
|
'bg_color' => $params->get('social_image_bg_color', '#1a1a2e'),
|
||||||
'text_color' => $params->get('social_image_text_color', '#ffffff'),
|
'text_color' => $params->get('social_image_text_color', '#ffffff'),
|
||||||
'font_size' => $params->get('social_image_font_size', 48),
|
'overlay' => $params->get('social_image_overlay', 'dark'),
|
||||||
'show_site_name' => (bool) $params->get('social_image_show_site_name', 1),
|
|
||||||
];
|
];
|
||||||
|
|
||||||
$result = SocialImageHelper::generate($title, $siteName, $config);
|
$backgroundPath = null;
|
||||||
|
$images = json_decode($article->images ?? '{}', true);
|
||||||
|
|
||||||
|
if (!empty($images['image_intro'])) {
|
||||||
|
$backgroundPath = JPATH_ROOT . '/' . ltrim($images['image_intro'], '/');
|
||||||
|
} elseif (!empty($images['image_fulltext'])) {
|
||||||
|
$backgroundPath = JPATH_ROOT . '/' . ltrim($images['image_fulltext'], '/');
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$imagePath = SocialImageHelper::generate($article->title, $siteName, $backgroundPath, $options);
|
||||||
|
$imageUrl = str_replace(JPATH_ROOT, Uri::root(true), str_replace('\\', '/', $imagePath));
|
||||||
|
|
||||||
|
$result = ['success' => true, 'image_url' => $imageUrl, 'image_path' => $imagePath];
|
||||||
|
} catch (\Throwable $e) {
|
||||||
|
$result = ['success' => false, 'error' => $e->getMessage()];
|
||||||
|
}
|
||||||
|
|
||||||
$this->app->setHeader('Content-Type', 'application/json; charset=utf-8');
|
$this->app->setHeader('Content-Type', 'application/json; charset=utf-8');
|
||||||
echo json_encode($result);
|
echo json_encode($result);
|
||||||
|
|||||||
@@ -17,234 +17,144 @@ use Joomla\CMS\Factory;
|
|||||||
|
|
||||||
class AnalyticsHelper
|
class AnalyticsHelper
|
||||||
{
|
{
|
||||||
/**
|
private static array $dayNames = ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'];
|
||||||
* Record or update engagement metrics for a post.
|
|
||||||
*/
|
|
||||||
public static function recordEngagement(int $postId, int $serviceId, string $serviceType, array $metrics): bool
|
|
||||||
{
|
|
||||||
$db = Factory::getDbo();
|
|
||||||
|
|
||||||
$postedAt = $metrics['posted_at'] ?? null;
|
public static function getPostingHeatmap(string $serviceType = '', int $days = 90): array
|
||||||
|
|
||||||
if ($postedAt) {
|
|
||||||
$timestamp = strtotime($postedAt);
|
|
||||||
$dayOfWeek = (int) date('w', $timestamp);
|
|
||||||
$hourOfDay = (int) date('G', $timestamp);
|
|
||||||
} else {
|
|
||||||
$dayOfWeek = 0;
|
|
||||||
$hourOfDay = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
$impressions = (int) ($metrics['impressions'] ?? 0);
|
|
||||||
$engagements = (int) ($metrics['engagements'] ?? 0);
|
|
||||||
$clicks = (int) ($metrics['clicks'] ?? 0);
|
|
||||||
$shares = (int) ($metrics['shares'] ?? 0);
|
|
||||||
|
|
||||||
$engagementRate = $impressions > 0
|
|
||||||
? round(($engagements / $impressions) * 100, 2)
|
|
||||||
: 0.00;
|
|
||||||
|
|
||||||
$query = $db->getQuery(true)
|
|
||||||
->select($db->quoteName('id'))
|
|
||||||
->from($db->quoteName('#__mokosuitecross_analytics'))
|
|
||||||
->where($db->quoteName('post_id') . ' = ' . $postId)
|
|
||||||
->where($db->quoteName('service_id') . ' = ' . $serviceId);
|
|
||||||
$db->setQuery($query);
|
|
||||||
$existingId = $db->loadResult();
|
|
||||||
|
|
||||||
if ($existingId) {
|
|
||||||
$query = $db->getQuery(true)
|
|
||||||
->update($db->quoteName('#__mokosuitecross_analytics'))
|
|
||||||
->set($db->quoteName('impressions') . ' = ' . $impressions)
|
|
||||||
->set($db->quoteName('engagements') . ' = ' . $engagements)
|
|
||||||
->set($db->quoteName('clicks') . ' = ' . $clicks)
|
|
||||||
->set($db->quoteName('shares') . ' = ' . $shares)
|
|
||||||
->set($db->quoteName('engagement_rate') . ' = ' . $engagementRate)
|
|
||||||
->where($db->quoteName('id') . ' = ' . (int) $existingId);
|
|
||||||
$db->setQuery($query);
|
|
||||||
$db->execute();
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
$record = (object) [
|
|
||||||
'post_id' => $postId,
|
|
||||||
'service_id' => $serviceId,
|
|
||||||
'service_type' => $serviceType,
|
|
||||||
'posted_at' => $postedAt,
|
|
||||||
'day_of_week' => $dayOfWeek,
|
|
||||||
'hour_of_day' => $hourOfDay,
|
|
||||||
'impressions' => $impressions,
|
|
||||||
'engagements' => $engagements,
|
|
||||||
'clicks' => $clicks,
|
|
||||||
'shares' => $shares,
|
|
||||||
'engagement_rate' => $engagementRate,
|
|
||||||
'created' => Factory::getDate()->toSql(),
|
|
||||||
];
|
|
||||||
|
|
||||||
$db->insertObject('#__mokosuitecross_analytics', $record);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get heatmap data as a 7x24 grid derived from actual post success data.
|
|
||||||
*/
|
|
||||||
public static function getHeatmapData(string $serviceType = '', int $days = 90): array
|
|
||||||
{
|
{
|
||||||
$db = Factory::getDbo();
|
$db = Factory::getDbo();
|
||||||
|
|
||||||
$query = $db->getQuery(true)
|
$query = $db->getQuery(true)
|
||||||
->select([
|
->select('DAYOFWEEK(' . $db->quoteName('p.posted_at') . ') - 1 AS dow')
|
||||||
'DAYOFWEEK(' . $db->quoteName('p.posted_at') . ') - 1 AS day_of_week',
|
->select('HOUR(' . $db->quoteName('p.posted_at') . ') AS hr')
|
||||||
'HOUR(' . $db->quoteName('p.posted_at') . ') AS hour_of_day',
|
->select('COUNT(*) AS cnt')
|
||||||
'COUNT(*) AS post_count',
|
|
||||||
])
|
|
||||||
->from($db->quoteName('#__mokosuitecross_posts', 'p'))
|
->from($db->quoteName('#__mokosuitecross_posts', 'p'))
|
||||||
->join('INNER', $db->quoteName('#__mokosuitecross_services', 's')
|
|
||||||
. ' ON ' . $db->quoteName('s.id') . ' = ' . $db->quoteName('p.service_id'))
|
|
||||||
->where($db->quoteName('p.status') . ' = ' . $db->quote('posted'))
|
->where($db->quoteName('p.status') . ' = ' . $db->quote('posted'))
|
||||||
->where($db->quoteName('p.posted_at') . ' IS NOT NULL')
|
->where($db->quoteName('p.posted_at') . ' IS NOT NULL');
|
||||||
->group('day_of_week')
|
|
||||||
->group('hour_of_day')
|
|
||||||
->order('day_of_week ASC')
|
|
||||||
->order('hour_of_day ASC');
|
|
||||||
|
|
||||||
if ($serviceType !== '') {
|
|
||||||
$query->where($db->quoteName('s.service_type') . ' = ' . $db->quote($serviceType));
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($days > 0) {
|
if ($days > 0) {
|
||||||
$cutoff = Factory::getDate('-' . $days . ' days')->toSql();
|
$since = Factory::getDate('now - ' . $days . ' days')->toSql();
|
||||||
$query->where($db->quoteName('p.posted_at') . ' >= ' . $db->quote($cutoff));
|
$query->where($db->quoteName('p.posted_at') . ' >= ' . $db->quote($since));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($serviceType !== '') {
|
||||||
|
$query->join('INNER', $db->quoteName('#__mokosuitecross_services', 's') . ' ON ' . $db->quoteName('s.id') . ' = ' . $db->quoteName('p.service_id'))
|
||||||
|
->where($db->quoteName('s.service_type') . ' = ' . $db->quote($serviceType));
|
||||||
|
}
|
||||||
|
|
||||||
|
$query->group('dow, hr')
|
||||||
|
->order('dow ASC, hr ASC');
|
||||||
|
|
||||||
$db->setQuery($query);
|
$db->setQuery($query);
|
||||||
$rows = $db->loadObjectList();
|
$rows = $db->loadObjectList();
|
||||||
|
|
||||||
$maxCount = 1;
|
|
||||||
|
|
||||||
foreach ($rows as $row) {
|
|
||||||
if ((int) $row->post_count > $maxCount) {
|
|
||||||
$maxCount = (int) $row->post_count;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$grid = [];
|
$grid = [];
|
||||||
|
|
||||||
for ($d = 0; $d < 7; $d++) {
|
for ($d = 0; $d < 7; $d++) {
|
||||||
for ($h = 0; $h < 24; $h++) {
|
$grid[$d] = array_fill(0, 24, 0);
|
||||||
$grid[$d][$h] = ['avg_rate' => 0.00, 'post_count' => 0];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach ($rows as $row) {
|
foreach ($rows as $row) {
|
||||||
$count = (int) $row->post_count;
|
$grid[(int) $row->dow][(int) $row->hr] = (int) $row->cnt;
|
||||||
$grid[(int) $row->day_of_week][(int) $row->hour_of_day] = [
|
|
||||||
'avg_rate' => round(($count / $maxCount) * 100, 2),
|
|
||||||
'post_count' => $count,
|
|
||||||
];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return $grid;
|
return $grid;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
public static function getBestTimes(string $serviceType = '', int $days = 90, int $limit = 5): array
|
||||||
* Get the best times to post ranked by post success frequency.
|
|
||||||
*/
|
|
||||||
public static function getBestTimes(string $serviceType = '', int $limit = 5): array
|
|
||||||
{
|
{
|
||||||
$db = Factory::getDbo();
|
$grid = self::getPostingHeatmap($serviceType, $days);
|
||||||
|
$slots = [];
|
||||||
|
|
||||||
$query = $db->getQuery(true)
|
foreach ($grid as $dow => $hours) {
|
||||||
->select([
|
foreach ($hours as $hour => $count) {
|
||||||
'DAYOFWEEK(' . $db->quoteName('p.posted_at') . ') - 1 AS day_of_week',
|
if ($count > 0) {
|
||||||
'HOUR(' . $db->quoteName('p.posted_at') . ') AS hour_of_day',
|
$slots[] = [
|
||||||
'COUNT(*) AS post_count',
|
'day' => self::$dayNames[$dow],
|
||||||
])
|
'hour' => $hour,
|
||||||
->from($db->quoteName('#__mokosuitecross_posts', 'p'))
|
'count' => $count,
|
||||||
->join('INNER', $db->quoteName('#__mokosuitecross_services', 's')
|
'label' => self::$dayNames[$dow] . ' ' . self::formatHour($hour),
|
||||||
. ' ON ' . $db->quoteName('s.id') . ' = ' . $db->quoteName('p.service_id'))
|
];
|
||||||
->where($db->quoteName('p.status') . ' = ' . $db->quote('posted'))
|
}
|
||||||
->where($db->quoteName('p.posted_at') . ' IS NOT NULL')
|
}
|
||||||
->group('day_of_week')
|
|
||||||
->group('hour_of_day')
|
|
||||||
->having('COUNT(*) >= 1')
|
|
||||||
->order('post_count DESC');
|
|
||||||
|
|
||||||
if ($serviceType !== '') {
|
|
||||||
$query->where($db->quoteName('s.service_type') . ' = ' . $db->quote($serviceType));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$db->setQuery($query, 0, $limit);
|
usort($slots, static fn($a, $b) => $b['count'] <=> $a['count']);
|
||||||
$rows = $db->loadAssocList();
|
|
||||||
|
|
||||||
$dayNames = ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'];
|
return \array_slice($slots, 0, $limit);
|
||||||
|
|
||||||
$results = [];
|
|
||||||
|
|
||||||
foreach ($rows as $row) {
|
|
||||||
$hour = (int) $row['hour_of_day'];
|
|
||||||
$ampm = $hour < 12 ? 'AM' : 'PM';
|
|
||||||
$hour12 = $hour % 12 ?: 12;
|
|
||||||
|
|
||||||
$results[] = [
|
|
||||||
'day_of_week' => (int) $row['day_of_week'],
|
|
||||||
'day_name' => $dayNames[(int) $row['day_of_week']],
|
|
||||||
'hour_of_day' => $hour,
|
|
||||||
'hour_label' => $hour12 . ':00 ' . $ampm,
|
|
||||||
'avg_rate' => round((float) $row['post_count'], 2),
|
|
||||||
'post_count' => (int) $row['post_count'],
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
return $results;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Get stats grouped by service type from actual post data.
|
|
||||||
*/
|
|
||||||
public static function getServiceBreakdown(int $days = 30): array
|
public static function getServiceBreakdown(int $days = 30): array
|
||||||
{
|
{
|
||||||
$db = Factory::getDbo();
|
$db = Factory::getDbo();
|
||||||
|
|
||||||
$query = $db->getQuery(true)
|
$query = $db->getQuery(true)
|
||||||
->select([
|
->select($db->quoteName('s.service_type'))
|
||||||
$db->quoteName('s.service_type'),
|
->select($db->quoteName('s.title', 'service_title'))
|
||||||
'COUNT(*) AS total_posts',
|
->select('COUNT(*) AS total')
|
||||||
'SUM(CASE WHEN ' . $db->quoteName('p.status') . ' = ' . $db->quote('posted') . ' THEN 1 ELSE 0 END) AS total_succeeded',
|
->select('SUM(CASE WHEN ' . $db->quoteName('p.status') . ' = ' . $db->quote('posted') . ' THEN 1 ELSE 0 END) AS success')
|
||||||
'SUM(CASE WHEN ' . $db->quoteName('p.status') . ' IN ('
|
->select('SUM(CASE WHEN ' . $db->quoteName('p.status') . ' = ' . $db->quote('failed') . ' THEN 1 ELSE 0 END) AS failed')
|
||||||
. $db->quote('failed') . ',' . $db->quote('permanently_failed')
|
|
||||||
. ') THEN 1 ELSE 0 END) AS total_failed',
|
|
||||||
])
|
|
||||||
->from($db->quoteName('#__mokosuitecross_posts', 'p'))
|
->from($db->quoteName('#__mokosuitecross_posts', 'p'))
|
||||||
->join('INNER', $db->quoteName('#__mokosuitecross_services', 's')
|
->join('INNER', $db->quoteName('#__mokosuitecross_services', 's') . ' ON ' . $db->quoteName('s.id') . ' = ' . $db->quoteName('p.service_id'));
|
||||||
. ' ON ' . $db->quoteName('s.id') . ' = ' . $db->quoteName('p.service_id'))
|
|
||||||
->group($db->quoteName('s.service_type'))
|
|
||||||
->order('total_posts DESC');
|
|
||||||
|
|
||||||
if ($days > 0) {
|
if ($days > 0) {
|
||||||
$cutoff = Factory::getDate('-' . $days . ' days')->toSql();
|
$since = Factory::getDate('now - ' . $days . ' days')->toSql();
|
||||||
$query->where($db->quoteName('p.created') . ' >= ' . $db->quote($cutoff));
|
$query->where($db->quoteName('p.created') . ' >= ' . $db->quote($since));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$query->group($db->quoteName(['s.service_type', 's.title']))
|
||||||
|
->order('total DESC');
|
||||||
|
|
||||||
$db->setQuery($query);
|
$db->setQuery($query);
|
||||||
$rows = $db->loadAssocList();
|
$rows = $db->loadObjectList();
|
||||||
|
|
||||||
foreach ($rows as &$row) {
|
$result = [];
|
||||||
$total = (int) $row['total_posts'];
|
|
||||||
$succeeded = (int) $row['total_succeeded'];
|
|
||||||
|
|
||||||
$row['total_posts'] = $total;
|
foreach ($rows as $row) {
|
||||||
$row['total_succeeded'] = $succeeded;
|
$total = (int) $row->total;
|
||||||
$row['total_failed'] = (int) $row['total_failed'];
|
$success = (int) $row->success;
|
||||||
$row['avg_engagement_rate'] = $total > 0 ? round(($succeeded / $total) * 100, 2) : 0;
|
$result[] = [
|
||||||
$row['total_impressions'] = 0;
|
'service_type' => $row->service_type,
|
||||||
$row['total_engagements'] = 0;
|
'service_title' => $row->service_title,
|
||||||
$row['total_clicks'] = 0;
|
'total' => $total,
|
||||||
$row['total_shares'] = 0;
|
'success' => $success,
|
||||||
|
'failed' => (int) $row->failed,
|
||||||
|
'success_rate' => $total > 0 ? round(($success / $total) * 100, 1) : 0.0,
|
||||||
|
'avg_per_day' => $days > 0 ? round($total / $days, 1) : 0.0,
|
||||||
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
return $rows;
|
return $result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function getServiceTypes(): array
|
||||||
|
{
|
||||||
|
$db = Factory::getDbo();
|
||||||
|
|
||||||
|
$query = $db->getQuery(true)
|
||||||
|
->select('DISTINCT ' . $db->quoteName('service_type'))
|
||||||
|
->from($db->quoteName('#__mokosuitecross_services'))
|
||||||
|
->where($db->quoteName('published') . ' = 1')
|
||||||
|
->order($db->quoteName('service_type') . ' ASC');
|
||||||
|
|
||||||
|
$db->setQuery($query);
|
||||||
|
|
||||||
|
return $db->loadColumn() ?: [];
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function formatHour(int $hour): string
|
||||||
|
{
|
||||||
|
if ($hour === 0) {
|
||||||
|
return '12:00 AM';
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($hour < 12) {
|
||||||
|
return $hour . ':00 AM';
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($hour === 12) {
|
||||||
|
return '12:00 PM';
|
||||||
|
}
|
||||||
|
|
||||||
|
return ($hour - 12) . ':00 PM';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -594,26 +594,13 @@ class CrossPostDispatcher
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Load service plugins using Joomla 5/6-compatible dispatcher pattern
|
// Load service plugins
|
||||||
PluginHelper::importPlugin('mokosuitecross');
|
PluginHelper::importPlugin('mokosuitecross');
|
||||||
$servicePlugins = [];
|
$plugins = [];
|
||||||
$event = new \Joomla\Event\Event('onMokoSuiteCrossGetServices', [$servicePlugins]);
|
Factory::getApplication()->triggerEvent('onMokoSuiteCrossGetServices', [&$plugins]);
|
||||||
|
|
||||||
try {
|
|
||||||
Factory::getApplication()->getDispatcher()->dispatch('onMokoSuiteCrossGetServices', $event);
|
|
||||||
} catch (\Throwable $e) {
|
|
||||||
// Dispatcher may not be available
|
|
||||||
}
|
|
||||||
|
|
||||||
$idx = 1;
|
|
||||||
|
|
||||||
while (isset($event[$idx])) {
|
|
||||||
$servicePlugins[] = $event[$idx];
|
|
||||||
$idx++;
|
|
||||||
}
|
|
||||||
|
|
||||||
$pluginMap = [];
|
$pluginMap = [];
|
||||||
foreach ($servicePlugins as $plugin) {
|
foreach ($plugins as $plugin) {
|
||||||
$pluginMap[$plugin->getServiceType()] = $plugin;
|
$pluginMap[$plugin->getServiceType()] = $plugin;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -626,7 +613,7 @@ class CrossPostDispatcher
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
$credentials = CredentialHelper::decrypt($post->credentials ?: '');
|
$credentials = json_decode($post->credentials, true) ?: [];
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$result = $plugin->deletePost($post->platform_post_id, $credentials);
|
$result = $plugin->deletePost($post->platform_post_id, $credentials);
|
||||||
|
|||||||
@@ -40,9 +40,9 @@ class MokoSuiteCrossHelper
|
|||||||
'posts' => 'COM_MOKOSUITECROSS_SUBMENU_POSTS',
|
'posts' => 'COM_MOKOSUITECROSS_SUBMENU_POSTS',
|
||||||
'services' => 'COM_MOKOSUITECROSS_SUBMENU_SERVICES',
|
'services' => 'COM_MOKOSUITECROSS_SUBMENU_SERVICES',
|
||||||
'templates' => 'COM_MOKOSUITECROSS_SUBMENU_TEMPLATES',
|
'templates' => 'COM_MOKOSUITECROSS_SUBMENU_TEMPLATES',
|
||||||
|
'logs' => 'COM_MOKOSUITECROSS_SUBMENU_LOGS',
|
||||||
'calendar' => 'COM_MOKOSUITECROSS_SUBMENU_CALENDAR',
|
'calendar' => 'COM_MOKOSUITECROSS_SUBMENU_CALENDAR',
|
||||||
'analytics' => 'COM_MOKOSUITECROSS_SUBMENU_ANALYTICS',
|
'analytics' => 'COM_MOKOSUITECROSS_SUBMENU_ANALYTICS',
|
||||||
'logs' => 'COM_MOKOSUITECROSS_SUBMENU_LOGS',
|
|
||||||
];
|
];
|
||||||
|
|
||||||
// Joomla 5+ toolbar submenu
|
// Joomla 5+ toolbar submenu
|
||||||
|
|||||||
@@ -91,16 +91,6 @@ class QueueProcessor
|
|||||||
$db->setQuery($query);
|
$db->setQuery($query);
|
||||||
$retryPosts = $db->loadObjectList() ?: [];
|
$retryPosts = $db->loadObjectList() ?: [];
|
||||||
|
|
||||||
// 3. Recover stale "posting" entries (stuck > 10 minutes)
|
|
||||||
$staleQuery = $db->getQuery(true)
|
|
||||||
->update($db->quoteName('#__mokosuitecross_posts'))
|
|
||||||
->set($db->quoteName('status') . ' = ' . $db->quote('queued'))
|
|
||||||
->set($db->quoteName('modified') . ' = ' . $db->quote($now))
|
|
||||||
->where($db->quoteName('status') . ' = ' . $db->quote('posting'))
|
|
||||||
->where($db->quoteName('modified') . ' < DATE_SUB(NOW(), INTERVAL 600 SECOND)');
|
|
||||||
$db->setQuery($staleQuery);
|
|
||||||
$db->execute();
|
|
||||||
|
|
||||||
$allPosts = array_merge($queuedPosts, $retryPosts);
|
$allPosts = array_merge($queuedPosts, $retryPosts);
|
||||||
|
|
||||||
foreach ($allPosts as $post) {
|
foreach ($allPosts as $post) {
|
||||||
|
|||||||
@@ -14,47 +14,52 @@ namespace Joomla\Component\MokoSuiteCross\Administrator\View\Calendar;
|
|||||||
defined('_JEXEC') or die;
|
defined('_JEXEC') or die;
|
||||||
|
|
||||||
use Joomla\CMS\Factory;
|
use Joomla\CMS\Factory;
|
||||||
use Joomla\CMS\Language\Text;
|
|
||||||
use Joomla\CMS\MVC\View\HtmlView as BaseHtmlView;
|
use Joomla\CMS\MVC\View\HtmlView as BaseHtmlView;
|
||||||
use Joomla\CMS\Router\Route;
|
|
||||||
use Joomla\CMS\Toolbar\ToolbarHelper;
|
use Joomla\CMS\Toolbar\ToolbarHelper;
|
||||||
use Joomla\Component\MokoSuiteCross\Administrator\Helper\MokoSuiteCrossHelper;
|
use Joomla\Component\MokoSuiteCross\Administrator\Helper\MokoSuiteCrossHelper;
|
||||||
|
|
||||||
class HtmlView extends BaseHtmlView
|
class HtmlView extends BaseHtmlView
|
||||||
{
|
{
|
||||||
|
public int $year;
|
||||||
public $ajaxUrl;
|
public int $month;
|
||||||
|
public array $events;
|
||||||
|
public $sidebar;
|
||||||
|
|
||||||
public function display($tpl = null): void
|
public function display($tpl = null): void
|
||||||
{
|
{
|
||||||
// ACL check
|
$input = Factory::getApplication()->input;
|
||||||
$canDo = MokoSuiteCrossHelper::getActions();
|
|
||||||
|
|
||||||
if (!$canDo->get('core.manage')) {
|
$this->year = $input->getInt('year', (int) date('Y'));
|
||||||
throw new \RuntimeException(Text::_('JERROR_ALERTNOAUTHOR'), 403);
|
$this->month = $input->getInt('month', (int) date('n'));
|
||||||
|
|
||||||
|
if ($this->month < 1 || $this->month > 12) {
|
||||||
|
$this->month = (int) date('n');
|
||||||
}
|
}
|
||||||
|
|
||||||
// Build AJAX URL for FullCalendar event source
|
if ($this->year < 2000 || $this->year > 2100) {
|
||||||
$this->ajaxUrl = Route::_('index.php?option=com_mokosuitecross&task=calendar.events&format=json', false);
|
$this->year = (int) date('Y');
|
||||||
|
}
|
||||||
|
|
||||||
|
$model = $this->getModel();
|
||||||
|
$this->events = $model->getEvents($this->year, $this->month);
|
||||||
|
|
||||||
$this->addToolbar();
|
$this->addToolbar();
|
||||||
|
|
||||||
MokoSuiteCrossHelper::addSubmenu('calendar');
|
MokoSuiteCrossHelper::addSubmenu('calendar');
|
||||||
|
$this->sidebar = \Joomla\CMS\HTML\Sidebar::render();
|
||||||
// Set document title
|
|
||||||
Factory::getApplication()->getDocument()->setTitle(
|
|
||||||
Text::_('COM_MOKOSUITECROSS_CALENDAR') . ' - ' . Text::_('COM_MOKOSUITECROSS')
|
|
||||||
);
|
|
||||||
|
|
||||||
parent::display($tpl);
|
parent::display($tpl);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function addToolbar(): void
|
protected function addToolbar(): void
|
||||||
{
|
{
|
||||||
ToolbarHelper::title(
|
$canDo = MokoSuiteCrossHelper::getActions();
|
||||||
Text::_('COM_MOKOSUITECROSS') . ' - ' . Text::_('COM_MOKOSUITECROSS_CALENDAR'),
|
|
||||||
'calendar'
|
ToolbarHelper::title('MokoSuiteCross -- Post Calendar', 'calendar');
|
||||||
);
|
ToolbarHelper::back('JTOOLBAR_BACK', 'index.php?option=com_mokosuitecross&view=dashboard');
|
||||||
ToolbarHelper::back('JTOOLBAR_BACK', Route::_('index.php?option=com_mokosuitecross&view=dashboard', false));
|
|
||||||
|
if ($canDo->get('core.admin')) {
|
||||||
|
ToolbarHelper::preferences('com_mokosuitecross');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ class HtmlView extends BaseHtmlView
|
|||||||
protected $serviceBreakdown;
|
protected $serviceBreakdown;
|
||||||
protected $dailyTrend;
|
protected $dailyTrend;
|
||||||
protected $topArticles;
|
protected $topArticles;
|
||||||
|
public $sidebar;
|
||||||
public $period;
|
public $period;
|
||||||
|
|
||||||
public function display($tpl = null): void
|
public function display($tpl = null): void
|
||||||
@@ -58,6 +58,7 @@ class HtmlView extends BaseHtmlView
|
|||||||
$this->addToolbar();
|
$this->addToolbar();
|
||||||
|
|
||||||
MokoSuiteCrossHelper::addSubmenu('dashboard');
|
MokoSuiteCrossHelper::addSubmenu('dashboard');
|
||||||
|
$this->sidebar = \Joomla\CMS\HTML\Sidebar::render();
|
||||||
|
|
||||||
parent::display($tpl);
|
parent::display($tpl);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,150 +12,118 @@
|
|||||||
defined('_JEXEC') or die;
|
defined('_JEXEC') or die;
|
||||||
|
|
||||||
use Joomla\CMS\Language\Text;
|
use Joomla\CMS\Language\Text;
|
||||||
use Joomla\CMS\Session\Session;
|
use Joomla\CMS\Router\Route;
|
||||||
|
|
||||||
/** @var \Joomla\Component\MokoSuiteCross\Administrator\View\Calendar\HtmlView $this */
|
/** @var \Joomla\Component\MokoSuiteCross\Administrator\View\Calendar\HtmlView $this */
|
||||||
|
|
||||||
$token = Session::getFormToken();
|
$year = $this->year;
|
||||||
$ajaxUrl = $this->ajaxUrl;
|
$month = $this->month;
|
||||||
|
$events = $this->events;
|
||||||
|
$today = date('Y-m-d');
|
||||||
|
|
||||||
|
$prevMonth = $month - 1;
|
||||||
|
$prevYear = $year;
|
||||||
|
|
||||||
|
if ($prevMonth < 1) {
|
||||||
|
$prevMonth = 12;
|
||||||
|
$prevYear--;
|
||||||
|
}
|
||||||
|
|
||||||
|
$nextMonth = $month + 1;
|
||||||
|
$nextYear = $year;
|
||||||
|
|
||||||
|
if ($nextMonth > 12) {
|
||||||
|
$nextMonth = 1;
|
||||||
|
$nextYear++;
|
||||||
|
}
|
||||||
|
|
||||||
|
$monthName = date('F', mktime(0, 0, 0, $month, 1, $year));
|
||||||
|
$daysInMonth = (int) date('t', mktime(0, 0, 0, $month, 1, $year));
|
||||||
|
$firstWeekday = ((int) date('N', mktime(0, 0, 0, $month, 1, $year))) - 1;
|
||||||
|
|
||||||
|
$statusClass = static function (string $status): string {
|
||||||
|
return match ($status) {
|
||||||
|
'posted' => 'bg-success',
|
||||||
|
'failed' => 'bg-danger',
|
||||||
|
default => 'bg-warning text-dark',
|
||||||
|
};
|
||||||
|
};
|
||||||
?>
|
?>
|
||||||
|
|
||||||
<style>
|
<div class="d-flex justify-content-between align-items-center mb-3">
|
||||||
#mokosuitecross-calendar {
|
<a href="<?php echo Route::_('index.php?option=com_mokosuitecross&view=calendar&year=' . $prevYear . '&month=' . $prevMonth); ?>"
|
||||||
max-width: 1100px;
|
class="btn btn-outline-secondary btn-sm">
|
||||||
margin: 0 auto;
|
<span class="icon-chevron-left" aria-hidden="true"></span>
|
||||||
}
|
<?php echo Text::_('COM_MOKOSUITECROSS_CALENDAR_PREV_MONTH'); ?>
|
||||||
.fc .fc-toolbar-title {
|
</a>
|
||||||
font-size: 1.4em;
|
<h3 class="mb-0"><?php echo htmlspecialchars($monthName . ' ' . $year); ?></h3>
|
||||||
}
|
<a href="<?php echo Route::_('index.php?option=com_mokosuitecross&view=calendar&year=' . $nextYear . '&month=' . $nextMonth); ?>"
|
||||||
.mokosuitecross-calendar-legend {
|
class="btn btn-outline-secondary btn-sm">
|
||||||
display: flex;
|
<?php echo Text::_('COM_MOKOSUITECROSS_CALENDAR_NEXT_MONTH'); ?>
|
||||||
gap: 1.5rem;
|
<span class="icon-chevron-right" aria-hidden="true"></span>
|
||||||
flex-wrap: wrap;
|
</a>
|
||||||
margin-bottom: 1rem;
|
|
||||||
}
|
|
||||||
.mokosuitecross-calendar-legend span {
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 0.35rem;
|
|
||||||
font-size: 0.875rem;
|
|
||||||
}
|
|
||||||
.mokosuitecross-calendar-legend .swatch {
|
|
||||||
display: inline-block;
|
|
||||||
width: 14px;
|
|
||||||
height: 14px;
|
|
||||||
border-radius: 3px;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|
||||||
<div class="mokosuitecross-calendar-legend">
|
|
||||||
<span><span class="swatch" style="background:#28a745;"></span> <?php echo Text::_('COM_MOKOSUITECROSS_STATUS_POSTED'); ?></span>
|
|
||||||
<span><span class="swatch" style="background:#007bff;"></span> <?php echo Text::_('COM_MOKOSUITECROSS_STATUS_SCHEDULED'); ?></span>
|
|
||||||
<span><span class="swatch" style="background:#ffc107;"></span> <?php echo Text::_('COM_MOKOSUITECROSS_STATUS_QUEUED'); ?></span>
|
|
||||||
<span><span class="swatch" style="background:#dc3545;"></span> <?php echo Text::_('COM_MOKOSUITECROSS_STATUS_FAILED'); ?></span>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="mokosuitecross-calendar"></div>
|
<div class="table-responsive">
|
||||||
|
<table class="table table-bordered">
|
||||||
|
<thead class="table-light">
|
||||||
|
<tr>
|
||||||
|
<th style="width:14.28%"><?php echo Text::_('MON'); ?></th>
|
||||||
|
<th style="width:14.28%"><?php echo Text::_('TUE'); ?></th>
|
||||||
|
<th style="width:14.28%"><?php echo Text::_('WED'); ?></th>
|
||||||
|
<th style="width:14.28%"><?php echo Text::_('THU'); ?></th>
|
||||||
|
<th style="width:14.28%"><?php echo Text::_('FRI'); ?></th>
|
||||||
|
<th style="width:14.28%"><?php echo Text::_('SAT'); ?></th>
|
||||||
|
<th style="width:14.28%"><?php echo Text::_('SUN'); ?></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<?php
|
||||||
|
$day = 1;
|
||||||
|
$started = false;
|
||||||
|
|
||||||
<script src="https://cdn.jsdelivr.net/npm/fullcalendar@6.1.15/index.global.min.js" integrity="sha384-B1OFx8Gy9GjPu8UbUyXbGQpzll9ubAUQ9agInFJ8NnD7nYG1u/CLR+Sqr5yifl4q" crossorigin="anonymous"></script>
|
while ($day <= $daysInMonth) : ?>
|
||||||
<script>
|
<tr>
|
||||||
document.addEventListener('DOMContentLoaded', function() {
|
<?php for ($col = 0; $col < 7; $col++) :
|
||||||
var calendarEl = document.getElementById('mokosuitecross-calendar');
|
if (!$started && $col < $firstWeekday) : ?>
|
||||||
var token = '<?php echo $token; ?>';
|
<td class="text-muted bg-light"> </td>
|
||||||
|
<?php
|
||||||
|
continue;
|
||||||
|
endif;
|
||||||
|
|
||||||
var calendar = new FullCalendar.Calendar(calendarEl, {
|
$started = true;
|
||||||
initialView: 'dayGridMonth',
|
|
||||||
headerToolbar: {
|
|
||||||
left: 'prev,next today',
|
|
||||||
center: 'title',
|
|
||||||
right: 'dayGridMonth,timeGridWeek,listWeek'
|
|
||||||
},
|
|
||||||
buttonText: {
|
|
||||||
today: '<?php echo Text::_('COM_MOKOSUITECROSS_CALENDAR_TODAY', true); ?>',
|
|
||||||
month: '<?php echo Text::_('COM_MOKOSUITECROSS_CALENDAR_MONTH', true); ?>',
|
|
||||||
week: '<?php echo Text::_('COM_MOKOSUITECROSS_CALENDAR_WEEK', true); ?>',
|
|
||||||
list: '<?php echo Text::_('COM_MOKOSUITECROSS_CALENDAR_LIST', true); ?>'
|
|
||||||
},
|
|
||||||
editable: true,
|
|
||||||
droppable: false,
|
|
||||||
navLinks: true,
|
|
||||||
dayMaxEvents: true,
|
|
||||||
eventSources: [{
|
|
||||||
url: '<?php echo $ajaxUrl; ?>',
|
|
||||||
method: 'GET',
|
|
||||||
failure: function() {
|
|
||||||
Joomla.renderMessages({
|
|
||||||
error: ['<?php echo Text::_('COM_MOKOSUITECROSS_CALENDAR_LOAD_ERROR', true); ?>']
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}],
|
|
||||||
eventClick: function(info) {
|
|
||||||
info.jsEvent.preventDefault();
|
|
||||||
if (info.event.url) {
|
|
||||||
window.location.href = info.event.url;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
eventDrop: function(info) {
|
|
||||||
var postId = info.event.id;
|
|
||||||
var status = info.event.extendedProps.status;
|
|
||||||
|
|
||||||
// Only allow rescheduling of scheduled or queued posts
|
if ($day > $daysInMonth) : ?>
|
||||||
if (status !== 'scheduled' && status !== 'queued') {
|
<td class="text-muted bg-light"> </td>
|
||||||
info.revert();
|
<?php
|
||||||
Joomla.renderMessages({
|
continue;
|
||||||
warning: ['<?php echo Text::_('COM_MOKOSUITECROSS_CALENDAR_CANNOT_RESCHEDULE', true); ?>']
|
endif;
|
||||||
});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var newDate = info.event.start.toISOString();
|
$dateKey = sprintf('%04d-%02d-%02d', $year, $month, $day);
|
||||||
|
$isToday = ($dateKey === $today);
|
||||||
var formData = new FormData();
|
$cellClass = $isToday ? 'border border-primary border-2 bg-primary bg-opacity-10' : '';
|
||||||
formData.append('post_id', postId);
|
$dayEvents = $events[$dateKey] ?? [];
|
||||||
formData.append('new_date', newDate);
|
?>
|
||||||
formData.append(token, '1');
|
<td class="<?php echo $cellClass; ?>" style="vertical-align: top; min-height: 80px;">
|
||||||
|
<div class="fw-bold mb-1<?php echo $isToday ? ' text-primary' : ''; ?>">
|
||||||
fetch('index.php?option=com_mokosuitecross&task=calendar.reschedule&format=json', {
|
<?php echo $day; ?>
|
||||||
method: 'POST',
|
<?php if ($isToday) : ?>
|
||||||
body: formData
|
<small class="text-primary"><?php echo Text::_('COM_MOKOSUITECROSS_CALENDAR_TODAY'); ?></small>
|
||||||
})
|
<?php endif; ?>
|
||||||
.then(function(response) { return response.json(); })
|
</div>
|
||||||
.then(function(data) {
|
<?php foreach ($dayEvents as $event) : ?>
|
||||||
if (data.success) {
|
<span class="badge <?php echo $statusClass($event->status); ?> mb-1 d-block text-truncate" style="max-width: 100%;"
|
||||||
// Update the event colour to scheduled
|
title="<?php echo htmlspecialchars(ucfirst($event->service_type) . ': ' . $event->article_title . ' (' . $event->status . ')'); ?>">
|
||||||
info.event.setProp('color', '#007bff');
|
<?php echo htmlspecialchars(ucfirst($event->service_type)); ?>:
|
||||||
info.event.setExtendedProp('status', 'scheduled');
|
<?php echo htmlspecialchars(mb_substr($event->article_title, 0, 20)); ?>
|
||||||
Joomla.renderMessages({
|
</span>
|
||||||
message: ['<?php echo Text::_('COM_MOKOSUITECROSS_CALENDAR_RESCHEDULE_SUCCESS', true); ?>']
|
<?php endforeach; ?>
|
||||||
});
|
</td>
|
||||||
} else {
|
<?php
|
||||||
info.revert();
|
$day++;
|
||||||
Joomla.renderMessages({
|
endfor; ?>
|
||||||
error: [data.error || '<?php echo Text::_('COM_MOKOSUITECROSS_CALENDAR_RESCHEDULE_ERROR', true); ?>']
|
</tr>
|
||||||
});
|
<?php endwhile; ?>
|
||||||
}
|
</tbody>
|
||||||
})
|
</table>
|
||||||
.catch(function() {
|
</div>
|
||||||
info.revert();
|
|
||||||
Joomla.renderMessages({
|
|
||||||
error: ['<?php echo Text::_('COM_MOKOSUITECROSS_CALENDAR_RESCHEDULE_ERROR', true); ?>']
|
|
||||||
});
|
|
||||||
});
|
|
||||||
},
|
|
||||||
eventDidMount: function(info) {
|
|
||||||
// Add tooltip with post details
|
|
||||||
var props = info.event.extendedProps;
|
|
||||||
var tip = info.event.title;
|
|
||||||
if (props.status) {
|
|
||||||
tip += ' [' + props.status + ']';
|
|
||||||
}
|
|
||||||
if (props.message) {
|
|
||||||
tip += '\n' + props.message;
|
|
||||||
}
|
|
||||||
info.el.setAttribute('title', tip);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
calendar.render();
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
|
|||||||
@@ -282,9 +282,9 @@ $queueProcessing = $componentParams->get('queue_processing', 'scheduler');
|
|||||||
class="list-group-item list-group-item-action">
|
class="list-group-item list-group-item-action">
|
||||||
<?php echo Text::_('COM_MOKOSUITECROSS_SUBMENU_LOGS'); ?>
|
<?php echo Text::_('COM_MOKOSUITECROSS_SUBMENU_LOGS'); ?>
|
||||||
</a>
|
</a>
|
||||||
<a href="<?php echo Route::_('index.php?option=com_mokosuitecross&view=calendar'); ?>"
|
<a href="<?php echo Route::_('index.php?option=com_mokosuitecross&view=analytics'); ?>"
|
||||||
class="list-group-item list-group-item-action">
|
class="list-group-item list-group-item-action">
|
||||||
<?php echo Text::_('COM_MOKOSUITECROSS_CALENDAR'); ?>
|
<?php echo Text::_('COM_MOKOSUITECROSS_SUBMENU_ANALYTICS'); ?>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="content" method="upgrade">
|
<extension type="plugin" group="content" method="upgrade">
|
||||||
<name>Content - MokoSuiteCross</name>
|
<name>Content - MokoSuiteCross</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -257,53 +257,6 @@ XML;
|
|||||||
$form->load($aiXml);
|
$form->load($aiXml);
|
||||||
$form->setFieldAttribute('mokosuitecross_ai_generate', 'description', $aiButtonHtml, 'attribs');
|
$form->setFieldAttribute('mokosuitecross_ai_generate', 'description', $aiButtonHtml, 'attribs');
|
||||||
}
|
}
|
||||||
// Social Image Generator button (#157)
|
|
||||||
$siParams = ComponentHelper::getParams('com_mokosuitecross');
|
|
||||||
$siEnabled = (bool) $siParams->get('social_image_enabled', 0);
|
|
||||||
|
|
||||||
if ($siEnabled && $articleId > 0) {
|
|
||||||
$siToken = Session::getFormToken();
|
|
||||||
$siUrl = Uri::base() . 'index.php?option=com_mokosuitecross&task=socialimage.generate&format=raw&article_id=' . $articleId . '&' . $siToken . '=1';
|
|
||||||
|
|
||||||
$siButtonHtml = '<div class="mb-3">'
|
|
||||||
. '<button type="button" id="mokosuitecross-si-btn" class="btn btn-sm btn-outline-success" onclick="mokosuitecrossSiGenerate()">'
|
|
||||||
. '<span class="icon-image" aria-hidden="true"></span> '
|
|
||||||
. \Joomla\CMS\Language\Text::_('COM_MOKOSUITECROSS_SOCIAL_IMAGE_GENERATE')
|
|
||||||
. '</button>'
|
|
||||||
. '<span id="mokosuitecross-si-status" class="ms-2 small"></span>'
|
|
||||||
. '<div id="mokosuitecross-si-preview" class="mt-2" style="display:none;">'
|
|
||||||
. '<img id="mokosuitecross-si-thumb" src="" alt="Social image preview" style="max-width:300px;border:1px solid #ccc;border-radius:4px;" />'
|
|
||||||
. '</div>'
|
|
||||||
. '</div>'
|
|
||||||
. '<script>'
|
|
||||||
. 'function mokosuitecrossSiGenerate(){'
|
|
||||||
. 'var btn=document.getElementById("mokosuitecross-si-btn");'
|
|
||||||
. 'var st=document.getElementById("mokosuitecross-si-status");'
|
|
||||||
. 'var pv=document.getElementById("mokosuitecross-si-preview");'
|
|
||||||
. 'var img=document.getElementById("mokosuitecross-si-thumb");'
|
|
||||||
. 'btn.disabled=true;st.textContent="' . \Joomla\CMS\Language\Text::_('COM_MOKOSUITECROSS_SOCIAL_IMAGE_GENERATING', true) . '";'
|
|
||||||
. 'pv.style.display="none";'
|
|
||||||
. 'fetch("' . $siUrl . '")'
|
|
||||||
. '.then(function(r){return r.json();})'
|
|
||||||
. '.then(function(d){'
|
|
||||||
. 'btn.disabled=false;'
|
|
||||||
. 'if(!d.success){st.textContent=d.error||"Error";return;}'
|
|
||||||
. 'st.textContent="' . \Joomla\CMS\Language\Text::_('COM_MOKOSUITECROSS_SOCIAL_IMAGE_GENERATED', true) . '";'
|
|
||||||
. 'img.src="' . Uri::root() . '"+d.image_url+"?t="+Date.now();'
|
|
||||||
. 'pv.style.display="block";'
|
|
||||||
. '})'
|
|
||||||
. '.catch(function(){btn.disabled=false;st.textContent="Request failed";});'
|
|
||||||
. '}'
|
|
||||||
. '</script>';
|
|
||||||
|
|
||||||
$siXml = '<?xml version="1.0"?>
|
|
||||||
<form><fields name="attribs"><fieldset name="mokosuitecross_share">
|
|
||||||
<field name="mokosuitecross_si_generate" type="note"
|
|
||||||
label="" description="" />
|
|
||||||
</fieldset></fields></form>';
|
|
||||||
$form->load($siXml);
|
|
||||||
$form->setFieldAttribute('mokosuitecross_si_generate', 'description', $siButtonHtml, 'attribs');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Cross-post history panel for existing articles
|
// Cross-post history panel for existing articles
|
||||||
|
|
||||||
@@ -451,7 +404,7 @@ XML;
|
|||||||
/**
|
/**
|
||||||
* Dispatch cross-post when an article is saved and published.
|
* Dispatch cross-post when an article is saved and published.
|
||||||
*/
|
*/
|
||||||
public function onContentAfterSave($event): void
|
public function onContentAfterSave(\Joomla\CMS\Event\Content\AfterSaveEvent $event): void
|
||||||
{
|
{
|
||||||
$context = $event->getContext();
|
$context = $event->getContext();
|
||||||
|
|
||||||
@@ -488,7 +441,7 @@ XML;
|
|||||||
/**
|
/**
|
||||||
* Dispatch cross-post when article state changes to published.
|
* Dispatch cross-post when article state changes to published.
|
||||||
*/
|
*/
|
||||||
public function onContentChangeState($event): void
|
public function onContentChangeState(\Joomla\CMS\Event\Content\ContentChangeStateEvent $event): void
|
||||||
{
|
{
|
||||||
$context = $event->getContext();
|
$context = $event->getContext();
|
||||||
|
|
||||||
@@ -535,19 +488,6 @@ XML;
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Respect first-publish-only: skip if article was previously posted
|
|
||||||
if ($params->get('post_on_first_publish_only', 0)) {
|
|
||||||
$existsQuery = $db->getQuery(true)
|
|
||||||
->select('COUNT(*)')
|
|
||||||
->from($db->quoteName('#__mokosuitecross_posts'))
|
|
||||||
->where($db->quoteName('article_id') . ' = ' . (int) $pk);
|
|
||||||
$db->setQuery($existsQuery);
|
|
||||||
|
|
||||||
if ((int) $db->loadResult() > 0) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$url = Uri::root() . 'index.php?option=com_content&view=article&id=' . $article->id;
|
$url = Uri::root() . 'index.php?option=com_content&view=article&id=' . $article->id;
|
||||||
|
|
||||||
if (!empty($article->catid)) {
|
if (!empty($article->catid)) {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - ActivityPub (Fediverse)</name>
|
<name>MokoSuiteCross - ActivityPub (Fediverse)</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Google Blogger</name>
|
<name>MokoSuiteCross - Google Blogger</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Bluesky</name>
|
<name>MokoSuiteCross - Bluesky</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Brevo (Sendinblue)</name>
|
<name>MokoSuiteCross - Brevo (Sendinblue)</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Constant Contact</name>
|
<name>MokoSuiteCross - Constant Contact</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - ConvertKit</name>
|
<name>MokoSuiteCross - ConvertKit</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Dev.to</name>
|
<name>MokoSuiteCross - Dev.to</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Discord</name>
|
<name>MokoSuiteCross - Discord</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Facebook / Meta</name>
|
<name>MokoSuiteCross - Facebook / Meta</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Ghost</name>
|
<name>MokoSuiteCross - Ghost</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Google Business Profile</name>
|
<name>MokoSuiteCross - Google Business Profile</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Google Chat</name>
|
<name>MokoSuiteCross - Google Chat</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Hashnode</name>
|
<name>MokoSuiteCross - Hashnode</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Instagram</name>
|
<name>MokoSuiteCross - Instagram</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-06-23</creationDate>
|
<creationDate>2026-06-23</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - LinkedIn</name>
|
<name>MokoSuiteCross - LinkedIn</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Mailchimp</name>
|
<name>MokoSuiteCross - Mailchimp</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Mastodon</name>
|
<name>MokoSuiteCross - Mastodon</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Matrix / Element</name>
|
<name>MokoSuiteCross - Matrix / Element</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Medium</name>
|
<name>MokoSuiteCross - Medium</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - MokoSuiteCalendar Events</name>
|
<name>MokoSuiteCross - MokoSuiteCalendar Events</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - MokoSuiteGallery</name>
|
<name>MokoSuiteCross - MokoSuiteGallery</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Nostr</name>
|
<name>MokoSuiteCross - Nostr</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Ntfy Push Notifications</name>
|
<name>MokoSuiteCross - Ntfy Push Notifications</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Pinterest</name>
|
<name>MokoSuiteCross - Pinterest</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Reddit</name>
|
<name>MokoSuiteCross - Reddit</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - RSS Feed</name>
|
<name>MokoSuiteCross - RSS Feed</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - SendGrid</name>
|
<name>MokoSuiteCross - SendGrid</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Slack</name>
|
<name>MokoSuiteCross - Slack</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Microsoft Teams</name>
|
<name>MokoSuiteCross - Microsoft Teams</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Telegram</name>
|
<name>MokoSuiteCross - Telegram</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Threads (Meta)</name>
|
<name>MokoSuiteCross - Threads (Meta)</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - TikTok</name>
|
<name>MokoSuiteCross - TikTok</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Tumblr</name>
|
<name>MokoSuiteCross - Tumblr</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - X / Twitter</name>
|
<name>MokoSuiteCross - X / Twitter</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Generic Webhook</name>
|
<name>MokoSuiteCross - Generic Webhook</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - WhatsApp Business</name>
|
<name>MokoSuiteCross - WhatsApp Business</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - WordPress</name>
|
<name>MokoSuiteCross - WordPress</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
<extension type="plugin" group="mokosuitecross" method="upgrade">
|
||||||
<name>MokoSuiteCross - Youtube</name>
|
<name>MokoSuiteCross - Youtube</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-06-23</creationDate>
|
<creationDate>2026-06-23</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="system" method="upgrade">
|
<extension type="plugin" group="system" method="upgrade">
|
||||||
<name>System - MokoSuiteCross</name>
|
<name>System - MokoSuiteCross</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="system" method="upgrade">
|
<extension type="plugin" group="system" method="upgrade">
|
||||||
<name>System - MokoSuiteCross Events</name>
|
<name>System - MokoSuiteCross Events</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="system" method="upgrade">
|
<extension type="plugin" group="system" method="upgrade">
|
||||||
<name>System - MokoSuiteCross Gallery</name>
|
<name>System - MokoSuiteCross Gallery</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="task" method="upgrade">
|
<extension type="plugin" group="task" method="upgrade">
|
||||||
<name>Task - MokoSuiteCross Queue Processor</name>
|
<name>Task - MokoSuiteCross Queue Processor</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="webservices" method="upgrade">
|
<extension type="plugin" group="webservices" method="upgrade">
|
||||||
<name>Web Services - MokoSuiteCross</name>
|
<name>Web Services - MokoSuiteCross</name>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
<extension type="package" method="upgrade">
|
<extension type="package" method="upgrade">
|
||||||
<name>MokoSuiteCross</name>
|
<name>MokoSuiteCross</name>
|
||||||
<packagename>mokosuitecross</packagename>
|
<packagename>mokosuitecross</packagename>
|
||||||
<version>01.13.04</version>
|
<version>01.10.00</version>
|
||||||
<creationDate>2026-05-28</creationDate>
|
<creationDate>2026-05-28</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
Reference in New Issue
Block a user