dhtmlxScheduler 与 PHP:Slim
本教程为您提供了使用 Slim 4 框架结合服务器端 REST API 构建基于 PHP 的 Scheduler 的基本步骤。
本教程使用 Slim Framework v4.x。 如果您正在使用较早的版本,请参考 Slim Framework v3.x 指南。
此外,还有其他平台和框架集成的教程可供参考:
- dhtmlxScheduler와 ASP.NET Core
- dhtmlxScheduler와 ASP.NET MVC
- dhtmlxScheduler와 Node.js
- dhtmlxScheduler와 PHP
- dhtmlxScheduler와 PHP:Laravel 연동하기
- dhtmlxScheduler와 SalesForce LWC 통합하기
- dhtmlxScheduler와 Ruby on Rails 연동하기
- dhtmlxScheduler와 dhtmlxConnector 연동하기
在开发 PHP 应用时,通常会采用现有框架,而不是从零开始搭建所有内容。
本指南将使用 Slim 4 框架以及服务器端的 REST API,数据存储采用 MySQL。CRUD 操作通过 PDO 实现,并设计为可灵活应用于其他框架。
现成的演示已在 GitHub 提供 供您参考。请按照以下步骤创建类似的应用。
完整源码可在 GitHub 获取。
步骤 1. 初始化项目
创建项目
首先,使用 Slim 4 框架的 skeleton application。
通过 Composer 创建应用:
$ composer create-project slim/slim-skeleton scheduler-slim-howto
$ cd scheduler-slim-howto/
步骤 2. 在页面中添加 Scheduler
下一步是在网页中放置一个 scheduler,主要分为两个简单的小步骤。
创建视图
在 app/templates 目录下创建一个 basic.html 文件:
<!doctype html>
<html>
<head>
<title> Getting started with dhtmlxScheduler</title>
<meta charSet="utf-8"/>
<script src="https://cdn.dhtmlx.com/scheduler/edge/dhtmlxscheduler.js"></script>
<link href="https://cdn.dhtmlx.com/scheduler/edge/dhtmlxscheduler.css"
rel="stylesheet" type="text/css" charSet="utf-8">
<style>
html, body{
margin:0px;
padding:0px;
height:100%;
overflow:hidden;
}
</style>
</head>
<body>
<div id="scheduler_here" className="dhx_cal_container">
<div className="dhx_cal_navline">
<div className="dhx_cal_prev_button"> </div>
<div className="dhx_cal_next_button"> </div>
<div className="dhx_cal_today_button"></div>
<div className="dhx_cal_date"></div>
<div className="dhx_cal_tab" name="day_tab"></div>
<div className="dhx_cal_tab" name="week_tab"></div>
<div className="dhx_cal_tab" name="month_tab"></div>
</div>
<div className="dhx_cal_header"></div>
<div className="dhx_cal_data"></div>
</div>
<script>
scheduler.config.xml_date="%Y-%m-%d %H:%i";
scheduler.init('scheduler_here', new Date(2019,0,20), "week");
scheduler.load("/events");
var dp = scheduler.createDataProcessor("/events");
dp.init(scheduler);
dp.setTransactionMode("REST"); // use to transfer data with REST
</script>
</body>
</html>
设置路由
新页面准备好后,需要通过浏览器访问。请在 app/routes.php 中添加一个路由:
$app->get('/', function (Request $request, Response $response) {
$payload = file_get_contents('../app/templates/basic.html');
$response->getBody()->write($payload);
return $response;
});
此时,运行应用即可在页面上看到 scheduler:

步骤 3. 准备数据库
scheduler 已经就位,接下来需要设置数据库并将其与应用连接。
创建数据库
可以使用您喜欢的 MySQL 客户端(如 phpMyAdmin)或命令行创建数据库。以下为创建数据库及日历事件表的 SQL:
CREATE DATABASE IF NOT EXISTS `scheduler`;
USE `scheduler`;
DROP TABLE IF EXISTS `events`;
CREATE TABLE `events` (
`id` int(11) AUTO_INCREMENT,
`start_date` datetime NOT NULL,
`end_date` datetime NOT NULL,
`text` varchar(255) DEFAULT NULL,
PRIMARY KEY (`id`)
) DEFAULT CHARSET="utf8;"
如需通过 MySQL 控制台导入,请将上述 SQL 保存为 dump.sql 文件并运行:
$ mysql -uuser -ppass scheduler < mysql_dump.sql
然后,打开 app/settings.php,添加数据库设置数组,并根据实际情况修改为您的数据库信息:
'pdo' => [
'engine' => 'mysql',
'host' => 'localhost',
'database' => 'scheduler',
'username' => 'user',
'password' => 'pass',
'charset' => 'utf8',
'collation' => 'utf8_unicode_ci',
'options' => [
PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
PDO::ATTR_EMULATE_PREPARES => true,
],
]
接下来,更新 app/dependencies.php,将 PDO 实例注入到应用容器:
// Inject a new instance of PDO into the container
$containerBuilder->addDefinitions([
PDO::class => function (ContainerInterface $container) {
$config = $container->get('settings')['pdo'];
$dsn = "{$config['engine']}:host="{$config["'host']};dbname="{$config["'database']};
charSet="{$config["'charset']}";
$username = $config['username'];
$password = $config['password'];
return new PDO($dsn, $username, $password, $config['options']);
},
]);
步骤 4. 加载数据
scheduler 已设置为从 "/events" 端点请求事件数据。现在需要 为该路由添加处理器以提供实际数据。
由于 scheduler 需要多个处理器,这里使用 Slim 4 的 route groups 进行组织。
打开 app/routes.php,为 "/events" 添加一个带 GET 操作的分组:
$app->group('/events', function ($group) {
$group->get('', function (Request $request, Response $response, array $args) {
$db = $this->get('PDO');
$queryText = 'SELECT * FROM `events`';
$params = $request->getQueryParams();
$query = $db->prepare($queryText);
$query->execute();
$result = $query->fetchAll();
$payload = json_encode($result);
$response->getBody()->write($payload);
return $response->withHeader('Content-Type', 'application/json');
});
});
只要数据库中有事件,它们就会显示在 scheduler 中。
动态加载
此时,scheduler 会在启动时加载所有事件记录。对于数据量较小的场景,这种方式没问题。但如果应用用于计划或预定且不会删除旧数据,数据量会逐渐增大,最终可能导致每次加载页面时都要请求大量数据。
动态加载可以避免这种情况,scheduler 会将当前显示的日期范围作为请求参数发送到服务器,服 务器只返回相关的记录。每当用户切换可见日期范围时,scheduler 会获取新的数据子集。
在客户端启用动态加载,只需使用 setLoadMode 方法,参数可为 "day"、"week" 或 "month"。例如:
scheduler.config.xml_date="%Y-%m-%d %H:%i";
scheduler.init("scheduler_here", new Date(2019, 0, 20), "week");
scheduler.setLoadMode("day");
scheduler.load("/events");
在服务器端,可以这样处理:
$app->group('/events', function ($group) {
$group->get('', function (Request $request, Response $response, array $args) {
$db = $this->get('PDO');
$queryText = 'SELECT * FROM `events`';
$params = $request->getQueryParams(); /*!*/
$queryParams = []; /*!*/
if (isset($params['from']) && isset($params['to'])) { /*!*/
$queryText .= " WHERE `end_date`>=? AND `start_date` < ?;"; /*!*/
$queryParams = [$params['from'], $params['to']]; /*!*/
} /*!*/
$query = $db->prepare($queryText);
$query->execute($queryParams); /*!*/
$result = $query->fetchAll();
$payload = json_encode($result);
$response->getBody()->write($payload);
return $response->withHeader('Content-Type', 'application/json');
});
});
步骤 5. 保存更改
实现后端处理程序
此时,调度器已经可以从后端获取数据。下一步是使其能够将更改保存回数据库。
客户端以 REST 模式运行,这意味着它会发送 POST、PUT 和 DELETE 请求来管理事件。 参考请求格式及调度器使用的所有路由。
为此,您需要定义一个控制器来处理数据模型上的操作,设置相应的路由,并在客户端启用数据保存功能。
回到 app/routes.php,为 "/events" 组添加 一个 POST 请求的处理程序。这将用于插入新事件:
$group->post('', function (Request $request, Response $response, array $args) {
$db = $this->get('PDO');
$body = $request->getParsedBody();
$queryText = 'INSERT INTO `events` SET
`start_date`=?,
`end_date`=?,
`text`=?';
$queryParams = [
$body['start_date'],
$body['end_date'],
$body['text']
];
$query = $db->prepare($queryText);
$query->execute($queryParams);
$result = [
'tid' => $db->lastInsertId(),
'action' => 'inserted'
];
$payload = json_encode($result);
$response->getBody()->write($payload);
return $response->withHeader('Content-Type', 'application/json');
});
当添加新任务时,其 ID 会通过响应对象的 tid 属性返回给客户端。
响应的 JSON 还可以包含其他属性,客户端处理程序可以访问这些属性。
同样,为 PUT 请求添加处理程序:
$group->put('/{id}', function (Request $request, Response $response, array $args) {
$db = $this->get('PDO');
$id = $request->getAttribute('route')->getArgument('id');
parse_str(file_get_contents("php://input"), $body);
$queryText = 'UPDATE `events` SET
`start_date`=?,
`end_date`=?,
`text`=?
WHERE `id`=?';
$queryParams = [
$body['start_date'],
$body['end_date'],
$body['text'],
$id
];
$query = $db->prepare($queryText);
$query->execute($queryParams);
$result = [
'action' => 'updated'
];
$payload = json_encode($result);
$response->getBody()->write($payload);
return $response->withHeader('Content-Type', 'application/json');
});
对于 DELETE 请求:
$group->delete('/{id}', function (Request $request, Response $response, array $args) {
$db = $this->get('PDO');
$id = $request->getAttribute('route')->getArgument('id');
$queryText = 'DELETE FROM `events` WHERE `id`=? ;';
$query = $db->prepare($queryText);
$query->execute([$id]);
$result = [
'action' => 'deleted'
];
$payload = json_encode($result);
$response->getBody()->write($payload);
return $response->withHeader('Content-Type', 'application/json');
});
启用客户端数据保存
最后,需配置客户端以与刚刚设置的 API 进行交互:
scheduler.config.xml_date="%Y-%m-%d %H:%i";
scheduler.init("scheduler_here", new Date(2019, 0, 20), "week");
scheduler.setLoadMode("day");
// 从后端获取数据
scheduler.load("/events");
// 将更改发送回后端
var dp = scheduler.createDataProcessor("/events"); dp.init(scheduler); /*!*/
// 配置数据交换模式
dp.setTransactionMode("REST"); /*!*/
重启应用后,调度器即可创建、删除和更新事件,并且所有更改在页面刷新后会被保留。

循环事件
若需支持循环事件(如每日重复),请在调度器页面引入相应扩展:
...
<body>
...
<script>
scheduler.plugins({
recurring: true /*!*/
});
scheduler.config.xml_date="%Y-%m-%d %H:%i";
scheduler.init("scheduler_here", new Date(2019, 0, 20), "week");
...
</script>
</body>
"events" 表需要额外的列来存储循环相关信息。以下 SQL 查询可创建支持循环事件的表:
CREATE DATABASE IF NOT EXISTS `scheduler`;
USE `scheduler`;
DROP TABLE IF EXISTS `events`;
CREATE TABLE `events` (
`id` int(11) AUTO_INCREMENT,
`start_date` datetime NOT NULL,
`end_date` datetime NOT NULL,
`text` varchar(255) DEFAULT NULL,
`event_pid` int(11) DEFAULT 0,
`event_length` bigint(20) unsigned DEFAULT 0,
`rec_type` varchar(25) DEFAULT '',
PRIMARY KEY (`id`)
) DEFAULT CHARSET="utf8;"
或者,也可以按如下方式更新现有 events 表:
ALTER TABLE `events` ADD COLUMN `event_pid` int(11) DEFAULT '0';
ALTER TABLE `events` ADD COLUMN `event_length` bigint(20) unsigned DEFAULT '0';
ALTER TABLE `events` ADD COLUMN `rec_type` varchar(25) DEFAULT '';