加载中...

yii_wiki_216_update-delete-model-with-cjuidialog-works-in-cgridview(通过CJuiDialog在CGridView中CRUD)


  1. /*****
  2. Update/delete model with CJuiDialog (works in CGridView)
  3.  
  4. http://www.yiiframework.com/wiki/216/update-delete-model-with-cjuidialog-works-in-cgridview
  5.  
  6. translated by php攻城师
  7.  
  8. http://blog.csdn.net/phpgcs
  9. Introduction
  10. Controller code
  11. Create action
  12. Update action
  13. Delete action
  14. View Code(for CGridView and simple links)
  15. Gii Code Generator
  16. Summary
  17.  
  18. ****/
  19.  
  20. /***
  21. Introduction
  22. ***/
  23. 我的方法基于 这篇wiki http://www.yiiframework.com/wiki/145/cjuidialog-for-create-new-model/
  24.  
  25. 这篇教程将向你介绍 如何 创建 Ajax dialog model 进行 CRUD 的操作。
  26. 适用于 简单的 links CGridView button column links 用最少的代码并且在JS关闭的情况下也可以完美的实现。
  27.  
  28. 现在已经有可用的扩展 extensions 了(本人开发的): EUpdateDialog
  29. http://www.yiiframework.com/extension/eupdatedialog/
  30. 但是需要注意的是, 对于(该扩展)最新的代码更新, 你应该仔细检查。
  31. 因为需要花费时间来更新2个相似的但是有些微不同的文档(extension this wiki), 我可能不会更新这篇教程(除非是很重要的更新)。
  32. 因此我的建议是 阅读这篇教程搞清楚工作原理, 然后检查 extension 文章和源代码。
  33. 带来的不方便,我感到很抱歉, 但是我认为花费的这些时间 如果用于更新 extension 本身会更好。
  34.  
  35. /****
  36.  
  37. << Controller code
  38.  
  39. <<Create action
  40. Update action
  41. Delete action
  42. View Code
  43. Gii Code Generator
  44. Summary
  45.  
  46. ***/
  47. Create action
  48.  
  49. public function actionCreate()
  50. {
  51. $model = new ModelName;
  52. if( isset( $_POST['ModelName'] ) )
  53. {
  54. $model->attributes = $_POST['ModelName'];
  55. if( $model->save() )
  56. {
  57. if( Yii::app()->request->isAjaxRequest )
  58. {
  59. // Stop jQuery from re-initialization
  60. Yii::app()->clientScript->scriptMap['jquery.js'] = false;
  61. echo CJSON::encode( array(
  62. 'status' => 'success',
  63. 'content' => 'ModelName successfully created',
  64. ));
  65. exit;
  66. }
  67. else
  68. $this->redirect( array( 'view', 'id' => $model->id ) );
  69. }
  70. }
  71. if( Yii::app()->request->isAjaxRequest )
  72. {
  73. // Stop jQuery from re-initialization
  74. Yii::app()->clientScript->scriptMap['jquery.js'] = false;
  75. echo CJSON::encode( array(
  76. 'status' => 'failure',
  77. 'content' => $this->renderPartial( '_form', array(
  78. 'model' => $model ), true, true ),
  79. ));
  80. exit;
  81. }
  82. else
  83. $this->render( 'create', array( 'model' => $model ) );
  84. }
  85.  
  86. 这样以来, 就可以简单地 save model 根据请求的类型来展示合适的内容。
  87. 重点是,
  88. 1 你需要用scriptMap 禁用 jQuery ,来阻止 重新初始化
  89. 2 如果你在 form 视图中使用 JS 代码的话,你要设置 renderPartial $processOutput parameter to true
  90.  
  91.  
  92. /****
  93. << Controller code
  94.  
  95. Create action
  96. <<Update action
  97. Delete action
  98. View Code
  99. Gii Code Generator
  100. Summary
  101.  
  102. ****/
  103.  
  104.  
  105. Update action create action 类似, 只有2个地方不同:
  106. 1 你需要先 load model
  107. 2 你需要 change success message.
  108.  
  109.  
  110. public function actionUpdate()
  111. {
  112. $model = $this->loadModel();
  113. if( isset( $_POST['ModelName'] ) )
  114. {
  115. $model->attributes = $_POST['ModelName'];
  116. if( $model->save() )
  117. {
  118. if( Yii::app()->request->isAjaxRequest )
  119. {
  120. // Stop jQuery from re-initialization
  121. Yii::app()->clientScript->scriptMap['jquery.js'] = false;
  122. echo CJSON::encode( array(
  123. 'status' => 'success',
  124. 'content' => 'ModelName successfully updated',
  125. ));
  126. exit;
  127. }
  128. else
  129. $this->redirect( array( 'view', 'id' => $model->id ) );
  130. }
  131. }
  132. if( Yii::app()->request->isAjaxRequest )
  133. {
  134. // Stop jQuery from re-initialization
  135. Yii::app()->clientScript->scriptMap['jquery.js'] = false;
  136. echo CJSON::encode( array(
  137. 'status' => 'failure',
  138. 'content' => $this->renderPartial( '_form', array(
  139. 'model' => $model ), true, true ),
  140. ));
  141. exit;
  142. }
  143. else
  144. $this->render( 'update', array( 'model' => $model ) );
  145. }
  146.  
  147.  
  148. /****
  149. << Controller code
  150.  
  151. Create action
  152. Update action
  153. <<Delete action
  154. View Code
  155. Gii Code Generator
  156. Summary
  157.  
  158. ****/
  159.  
  160. 不像默认的Yii 展示JS确认对话框那样的动作 ,这里展示了普通的 HTML 表单;这种方式 的好处在于, 即使JS 功能被禁用 了, 你仍然可以 delete model
  161.  
  162. public function actionDelete()
  163. {
  164. $model = $this->loadModel();
  165. if( Yii::app()->request->isAjaxRequest )
  166. {
  167. // Stop jQuery from re-initialization
  168. Yii::app()->clientScript->scriptMap['jquery.js'] = false;
  169. if( isset( $_POST['action'] ) && $_POST['action'] == 'confirmDelete' )
  170. {
  171. $model->delete();
  172. echo CJSON::encode( array(
  173. 'status' => 'success',
  174. 'content' => 'Deleted succussfully',
  175. ));
  176. exit;
  177. }
  178. else if( isset( $_POST['action'] ) )
  179. {
  180. echo CJSON::encode( array(
  181. 'status' => 'canceled',
  182. 'content' => 'Deletion canceled',
  183. ));
  184. exit;
  185. }
  186. else
  187. {
  188. echo CJSON::encode( array(
  189. 'status' => 'failure',
  190. 'content' => $this->renderPartial( 'delete', array(
  191. 'model' => $model ), true, true ),
  192. ));
  193. exit;
  194. }
  195. }
  196. else
  197. {
  198. if( isset( $_POST['confirmDelete'] ) )
  199. {
  200. $model->delete();
  201. $this->redirect( array( 'admin' ) );
  202. }
  203. else if( isset( $_POST['denyDelete'] ) )
  204. $this->redirect( array( 'view', 'id' => $model->id ) );
  205. else
  206. $this->render( 'delete', array( 'model' => $model ) );
  207. }
  208. }
  209.  
  210. 这个动作 检查是否是Ajax 请求,
  211. 如果是, 会检查 用户是 确认/拒绝 了删除model的请求,如果两者都不是, 那就再渲染一个 confirmation form Delete Confirmation 视图。
  212. 这个 Delete Confirmation 视图 需要至少 2个提交按钮(确认, 拒绝)。
  213.  
  214. 如果 浏览器禁用了 JS Delete Confirmation 视图 将会正常地被渲染, 从而同样可以达到delete model 的目的。
  215.  
  216. // You need to have a form in your delete view file!
  217. <?php $form = $this->beginWidget( 'CActiveForm', array(
  218. 'id' => 'location-delete-form',
  219. 'enableAjaxValidation' => false,
  220. 'focus' => '#confirmDelete',
  221. )); ?>
  222. <div class="buttons">
  223. <?php
  224. echo CHtml::submitButton( 'Yes', array( 'name' => 'confirmDelete',
  225. 'id' => 'confirmDelete' ) );
  226. echo CHtml::submitButton( 'No', array( 'name' => 'denyDelete' ) );
  227. ?>
  228. <?php
  229. /* !!! Or you can use jQuery UI buttons, makes no difference !!!
  230. $this->widget( 'zii.widgets.jui.CJuiButton', array(
  231. 'name' => 'confirmDelete',
  232. 'caption' => 'Yes',
  233. ));
  234. $this->widget( 'zii.widgets.jui.CJuiButton', array(
  235. 'name' => 'denyDelete',
  236. 'caption' => 'No',
  237. ));*/
  238. ?>
  239. </div>
  240. <?php $this->endWidget(); ?>
  241.  
  242. /****
  243. translated by php攻城师
  244.  
  245. http://blog.csdn.net/phpgcs
  246.  
  247.  
  248. Controller code
  249.  
  250. Create action
  251. Update action
  252. Delete action
  253.  
  254. << View Code
  255. Gii Code Generator
  256. Summary
  257.  
  258. ****/
  259.  
  260.  
  261. 如果你要在 CGridView widget 里面使用 上面的功能, 代码如下:
  262.  
  263. <?php $this->widget( 'zii.widgets.grid.CGridView', array(
  264. // # your widget settings here #
  265. 'columns' => array(
  266. // # your columns #
  267. array(
  268. 'class' => 'CButtonColumn',
  269. 'header' => 'Action',
  270. 'deleteButtonUrl' => 'Yii::app()->createUrl(
  271. "/admin/location/delete",
  272. array( "id" => $data->primaryKey ) )',
  273. 'buttons' => array(
  274. 'delete' => array(
  275. 'click' => "function( e ){
  276. e.preventDefault();
  277. $( '#update-dialog' ).children( ':eq(0)' ).empty(); // Stop auto POST
  278. updateDialog( $( this ).attr( 'href' ) );
  279. $( '#update-dialog' )
  280. .dialog( { title: 'Delete confirmation' } )
  281. .dialog( 'open' ); }",
  282. ),
  283. 'update' => array(
  284. 'click' => "function( e ){
  285. e.preventDefault();
  286. $( '#update-dialog' ).children( ':eq(0)' ).empty(); // Stop auto POST
  287. updateDialog( $( this ).attr( 'href' ) );
  288. $( '#update-dialog' )
  289. .dialog( { title: 'Update' } )
  290. .dialog( 'open' ); }",
  291. ),
  292. ),
  293. ),
  294. ),
  295. )); ?>
  296.  
  297. 这段代码 更改 delete 按钮 重定向到 delete confirmation 从而可以在无JS 的情况下工作。
  298. 也将 delete update 按钮的点击属性替换为 自定义的函数, 这个函数 禁止了 link 的默认行为 清除了 dialog 的内容, 链接url 给了即将打开的dialog 最后打开dialog
  299.  
  300.  
  301.  
  302. <?php
  303. $this->beginWidget( 'zii.widgets.jui.CJuiDialog', array(
  304. 'id' => 'update-dialog',
  305. 'options' => array(
  306. 'title' => 'Dialog',
  307. 'autoOpen' => false,
  308. 'modal' => true,
  309. 'width' => 550,
  310. 'resizable' => false,
  311. ),
  312. )); ?>
  313. <div class="update-dialog-content"></div>
  314. <?php $this->endWidget(); ?>
  315.  
  316. 这段代码 初始化了CJuiDialog 从而可以在我们需要的时候随时调用。
  317.  
  318.  
  319.  
  320. <?php
  321. $updateJS = CHtml::ajax( array(
  322. 'url' => "js:url",
  323. 'data' => "js:form.serialize() + action",
  324. 'type' => 'post',
  325. 'dataType' => 'json',
  326. 'success' => "function( data )
  327. {
  328. if( data.status == 'failure' )
  329. {
  330. $( '#update-dialog div.update-dialog-content' ).html( data.content );
  331. $( '#update-dialog div.update-dialog-content form input[type=submit]' )
  332. .die() // Stop from re-binding event handlers
  333. .live( 'click', function( e ){ // Send clicked button value
  334. e.preventDefault();
  335. updateDialog( false, $( this ).attr( 'name' ) );
  336. });
  337. }
  338. else
  339. {
  340. $( '#update-dialog div.update-dialog-content' ).html( data.content );
  341. if( data.status == 'success' ) // Update all grid views on success
  342. {
  343. $( 'div.grid-view' ).each( function(){ // Change the selector if you use different class or element
  344. $.fn.yiiGridView.update( $( this ).attr( 'id' ) );
  345. });
  346. }
  347. setTimeout( \"$( '#update-dialog' ).dialog( 'close' ).children( ':eq(0)' ).empty();\", 1000 );
  348. }
  349. }"
  350. )); ?>
  351.  
  352. 这段代码将 ajax 保存在php变量中
  353.  
  354. 如果请求失败, 意味着 what we have a form to display it adds retrieved code to dialog,对 submint 按钮 去除所有的 live event handlers
  355. (否则你下此打开 dialog 将会提交 2个请求。。然后3个。。等等)然后再重新把 live event handlers 赋给 submint 按钮。
  356.  
  357. 然后用户点击了 submit 按钮,
  358. Then user clicks submit button, it stops the form from submiting and sends his name attribute to update function.
  359.  
  360. 如果返回的 status 不是 'failure', 将会 把受到的数据 展示在dialog中.
  361. 进而如果 status 'success' 意味着 model 已经成功地 deleted/updated/created 那么将会更新所有的 grid view widgets
  362. 最后增加一个 timeout 函数关闭 并清除 dialog.
  363.  
  364. <?php
  365. Yii::app()->clientScript->registerScript( 'updateDialog', "
  366. function updateDialog( url, act )
  367. {
  368. var action = '';
  369. var form = $( '#update-dialog div.update-dialog-content form' );
  370. if( url == false )
  371. {
  372. action = '&action=' + act;
  373. url = form.attr( 'action' );
  374. }
  375. {$updateJS}
  376. }" ); ?>
  377.  
  378. 这个函数 完成了所有的更新。
  379. 首先,它设置了 需要的 变量, 然后检查url 参数是否提供了。
  380. 如果 提供了, 将展示合适的 form
  381. 如果没有提供, url false 意味着 form 已经被提交, 而它将 设置 action varialbe form 得到 url 并做出合适的 ajax 请求。
  382.  
  383. <?php
  384. Yii::app()->clientScript->registerScript( 'updateDialogCreate', "
  385. jQuery( function($){
  386. $( 'a.update-dialog-create' ).bind( 'click', function( e ){
  387. e.preventDefault();
  388. $( '#update-dialog' ).children( ':eq(0)' ).empty();
  389. updateDialog( $( this ).attr( 'href' ) );
  390. $( '#update-dialog' )
  391. .dialog( { title: 'Create' } )
  392. .dialog( 'open' );
  393. });
  394. });
  395. " );
  396. ?>
  397.  
  398. 为了 links 添加 dialog 功能 你只需要添加 把上面这段 script 即可。(这段script 为所有的 links a.update-dialog-create 绑定了处理程序的点击事件)
  399.  
  400.  
  401.  
  402. /****
  403. translated by php攻城师
  404.  
  405. http://blog.csdn.net/phpgcs
  406.  
  407.  
  408. Controller code
  409.  
  410. Create action
  411. Update action
  412. Delete action
  413.  
  414. View Code
  415. << Gii Code Generator
  416. Summary
  417.  
  418. ****/
  419.  
  420. 默认的 gii 生成的代码 需要做一些修改。
  421.  
  422. public function loadModel()
  423. {
  424. if( $this->_model === null )
  425. {
  426. if( isset( $_GET['id'] ) )
  427. $this->_model = ModelName::model()->findByPk( (int)$_GET['id'] );
  428. if( $this->_model === null )
  429. throw new CHttpException( 404, 'The requested page does not exist.' );
  430. }
  431. return $this->_model;
  432. }


还没有评论.