[ Index ] |
PHP Cross Reference of BuddyPress |
[Summary view] [Print] [Text view]
1 /* global bpGroupManageMembersSettings, _, Backbone */ 2 /* @version 5.0.0 */ 3 4 ( function( wp, bp, $ ) { 5 6 // Bail if not set 7 if ( typeof bpGroupManageMembersSettings === 'undefined' ) { 8 return; 9 } 10 11 // Copy useful WP Objects into BP. 12 _.extend( bp, _.pick( wp, 'Backbone', 'template' ) ); 13 14 bp.Models = bp.Models || {}; 15 bp.Collections = bp.Collections || {}; 16 bp.Views = bp.Views || {}; 17 18 /** 19 * Model for the Member of the displayed group. 20 */ 21 bp.Models.groupMember = Backbone.Model.extend( { 22 defaults: { 23 id: 0, 24 name: '', 25 avatar_urls : {}, 26 is_admin: false, 27 is_banned: false, 28 is_confirmed: false, 29 is_mod: false, 30 link: '' 31 }, 32 options : { 33 path: bpGroupManageMembersSettings.path, 34 type: 'POST', 35 data: {}, 36 dataType: 'json' 37 }, 38 39 initialize: function() { 40 // Make sure to reset data & path on model's sync. 41 this.on( 'sync', this.resetRequestOptions, this ); 42 }, 43 44 resetRequestOptions: function() { 45 this.options.data = {}; 46 this.options.path = bpGroupManageMembersSettings.path; 47 }, 48 49 sync: function( method, model, options ) { 50 options = options || {}; 51 options.context = this; 52 var data = options.data || {}; 53 this.options.path = this.options.path.concat( '/' + model.get( 'id' ) ); 54 55 _.extend( options, this.options ); 56 _.extend( options.data, data ); 57 58 if ( 'delete' === method || 'update' === method ) { 59 if ( 'delete' === method ) { 60 options.headers = { 'X-HTTP-Method-Override': 'DELETE' }; 61 } else { 62 options.headers = { 'X-HTTP-Method-Override': 'PUT' }; 63 } 64 65 return wp.apiRequest( options ); 66 } 67 }, 68 69 parse: function( response ) { 70 if ( _.isArray( response ) ) { 71 response = _.first( response ); 72 } 73 74 return response; 75 } 76 } ); 77 78 /** 79 * Collection for the Members of the displayed group. 80 */ 81 bp.Collections.groupMembers = Backbone.Collection.extend( { 82 model: bp.Models.groupMember, 83 options : { 84 path: bpGroupManageMembersSettings.path, 85 type: 'GET', 86 data: {}, 87 dataType: 'json' 88 }, 89 90 initialize: function() { 91 // Make sure to reset data on collection's reset. 92 this.on( 'reset', function() { 93 this.options.data = {}; 94 }, this ); 95 }, 96 97 sync: function( method, collection, options ) { 98 options = options || {}; 99 options.context = this; 100 var data = options.data || {}; 101 102 _.extend( options, this.options ); 103 _.extend( options.data, data ); 104 105 if ( 'read' === method ) { 106 var self = this, success = options.success; 107 options.success = function( data, textStatus, request ) { 108 if ( ! _.isUndefined( request ) ) { 109 self.totalPages = parseInt( request.getResponseHeader( 'X-WP-TotalPages' ), 10 ); 110 self.totalGroupMembers = parseInt( request.getResponseHeader( 'X-WP-Total' ), 10 ); 111 } 112 113 self.currentPage = options.data.page || 1; 114 115 if ( success ) { 116 return success.apply( this, arguments ); 117 } 118 }; 119 120 return wp.apiRequest( options ); 121 } 122 } 123 } ); 124 125 // Extend wp.Backbone.View with .prepare(). 126 bp.View = bp.View || bp.Backbone.View.extend( { 127 prepare: function() { 128 if ( ! _.isUndefined( this.model ) && _.isFunction( this.model.toJSON ) ) { 129 return this.model.toJSON(); 130 } else { 131 return {}; 132 } 133 } 134 } ); 135 136 bp.Views.GroupMemberUpdatingInfo = bp.View.extend( { 137 tagName: 'p', 138 template : bp.template( 'bp-manage-members-updating' ), 139 140 initialize: function() { 141 this.model = new Backbone.Model( { 142 type: this.options.value 143 } ); 144 } 145 } ); 146 147 bp.Views.GroupMemberErrorInfo = bp.View.extend( { 148 tagName: 'p', 149 template : bp.template( 'bp-manage-members-error' ), 150 151 initialize: function() { 152 this.model = new Backbone.Model( { 153 message: this.options.value 154 } ); 155 } 156 } ); 157 158 bp.Views.GroupsMembersLabel = bp.Views.GroupMemberUpdatingInfo.extend( { 159 tagName: 'label', 160 template: bp.template( 'bp-manage-members-label' ) 161 } ); 162 163 bp.Views.GroupRolesDropDown = bp.View.extend( { 164 tagName: 'select', 165 filters: _.extend( { all: { name: bpGroupManageMembersSettings.strings.allMembers } }, bpGroupManageMembersSettings.roles ), 166 167 events: { 168 change: 'change' 169 }, 170 171 initialize: function() { 172 if ( this.options.omits ) { 173 this.filters = _.omit( this.filters, this.options.omits ); 174 } 175 176 // Build `<option>` elements. 177 this.$el.html( _.chain( this.filters ).map( function( filter, value ) { 178 var optionOutput = $( '<option></option>' ).val( value ).html( filter.name )[0]; 179 180 if ( this.options.currentRole && value === this.options.currentRole ) { 181 return { 182 el: $( optionOutput ).prop( 'selected', true ) 183 }; 184 } else { 185 return { 186 el: optionOutput 187 }; 188 } 189 }, this ).pluck( 'el' ).value() ); 190 }, 191 192 change: function( event ) { 193 var role = $( event.target ).val(), queryArgs = { roles: [ role ] }; 194 195 if ( ! this.collection ) { 196 return; 197 } 198 199 if ( 'all' === role ) { 200 // Unset the current role. 201 this.collection.currentRole = ''; 202 203 queryArgs = { 'exclude_admins': false }; 204 } else { 205 // Set the current role. 206 this.collection.currentRole = role; 207 } 208 209 // Reset the current page. 210 this.collection.currentPage = 1; 211 212 queryArgs.page = 1; 213 $( '#manage-members-search' ).val( '' ); 214 215 this.collection.fetch( { 216 data: queryArgs, 217 reset: true 218 } ); 219 } 220 } ); 221 222 bp.Views.GroupMembersSearch = bp.View.extend( { 223 className: 'bp-dir-search-form', 224 tagName: 'form', 225 template: bp.template( 'bp-manage-members-search' ), 226 227 events: { 228 'click #manage-members-search-submit' : 'searchMember' 229 }, 230 231 searchMember: function( event ) { 232 event.preventDefault(); 233 234 var searchTerms = $( '#manage-members-search' ).val(), 235 queryArgs = _.extend( this.collection.options.data, { search: searchTerms, page: 1 } ); 236 237 // Reset the current page. 238 this.collection.currentPage = 1; 239 240 if ( ! this.collection.currentRole ) { 241 queryArgs.exclude_admins = false; 242 } else { 243 queryArgs.roles = [ this.collection.currentRole ]; 244 } 245 246 this.collection.fetch( { 247 data: queryArgs, 248 reset: true 249 } ); 250 } 251 } ); 252 253 bp.Views.GroupsMembersPagination = bp.View.extend( { 254 className: 'bp-pagination', 255 template: bp.template( 'bp-manage-members-paginate' ), 256 257 events: { 258 'click .group-members-paginate-button' : 'queryPage' 259 }, 260 261 initialize: function() { 262 this.collection.on( 'reset', this.setPagination, this ); 263 }, 264 265 setPagination: function( collection ) { 266 var attributes = _.pick( collection, [ 'currentPage', 'totalGroupMembers', 'totalPages' ] ); 267 268 if ( attributes.totalPages > 1 ) { 269 attributes.nextPage = attributes.currentPage + 1; 270 attributes.prevPage = attributes.currentPage - 1; 271 } 272 273 this.model = new Backbone.Model( attributes ); 274 this.render(); 275 }, 276 277 queryPage: function( event ) { 278 event.preventDefault(); 279 280 var page = $( event.currentTarget ).data( 'page' ), 281 searchTerms = $( '#manage-members-search' ).val(), 282 queryArgs = _.extend( this.collection.options.data, { search: searchTerms, page: page } ); 283 284 if ( ! this.collection.currentRole ) { 285 queryArgs.exclude_admins = false; 286 } else { 287 queryArgs.roles = [ this.collection.currentRole ]; 288 } 289 290 this.collection.fetch( { 291 data: queryArgs, 292 reset: true 293 } ); 294 } 295 } ); 296 297 bp.Views.GroupMembersNoMatches = bp.View.extend( { 298 tagName: 'tr', 299 template : bp.template( 'bp-manage-members-empty-row' ) 300 } ); 301 302 bp.Views.GroupMembersListRow = bp.View.extend( { 303 tagName: 'tr', 304 template : bp.template( 'bp-manage-members-row' ), 305 306 events: { 307 'click .group-member-actions a' : 'doMemberAction', 308 'change .group-member-edit select' : 'editMemberRole' 309 }, 310 311 initialize: function() { 312 var roleProps = [ 'is_admin', 'is_banned', 'is_confirmed', 'is_mod' ], 313 self = this; 314 315 _.each( bpGroupManageMembersSettings.roles, function( props ) { 316 if ( _.isMatch( self.model.attributes, _.pick( props, roleProps ) ) ) { 317 self.model.set( 'role', _.pick( props, ['id', 'name'] ), { silent: true } ); 318 } 319 } ); 320 321 this.model.collection.on( 'reset', this.clearRow, this ); 322 }, 323 324 clearRow: function() { 325 this.views.view.remove(); 326 }, 327 328 renderEditForm: function() { 329 var userId = this.model.get( 'id' ); 330 331 this.render(); 332 333 this.views.set( '#edit-group-member-' + userId, [ 334 new bp.Views.GroupsMembersLabel( { value: userId, attributes: { 'for': 'group-member' + userId + '-role' } } ), 335 new bp.Views.GroupRolesDropDown( { id: 'group-member' + userId + '-role', omits: [ 'all', 'banned' ], currentRole: this.model.get( 'role' ).id } ).render() 336 ] ); 337 }, 338 339 resetRow: function() { 340 this.model.set( 'editing', false ); 341 342 return this.render(); 343 }, 344 345 getRoleObject: function( roleId ) { 346 var roles = bpGroupManageMembersSettings.roles; 347 348 if ( _.isUndefined( roles[ roleId ] ) ) { 349 return {}; 350 } 351 352 return _.extend( 353 { role: _.pick( roles[ roleId ], ['id', 'name'] ) }, 354 _.pick( roles[ roleId ], [ 'is_admin', 'is_banned', 'is_confirmed', 'is_mod' ] ) 355 ); 356 }, 357 358 doMemberAction: function( event ) { 359 event.preventDefault(); 360 361 var action = $( event.target ).data( 'action' ), self = this; 362 363 if ( 'edit' === action ) { 364 this.model.set( 'editing', true ); 365 return this.renderEditForm(); 366 367 } else if ( 'abort' === action ) { 368 return this.resetRow(); 369 370 } else if ( 'ban' === action || 'unban' === action ) { 371 var newRole = ( 'ban' === action ) ? 'banned' : 'member', roleObject = this.getRoleObject( newRole ); 372 373 if ( ! roleObject ) { 374 return this.resetRow(); 375 } else { 376 this.model.set( 'managingBan', true ); 377 this.render(); 378 } 379 380 // Display user feedback. 381 this.views.set( '#edit-group-member-' + this.model.get( 'id' ), new bp.Views.GroupMemberUpdatingInfo( { value: action } ).render() ); 382 383 // Update Group member's role. 384 this.model.save( roleObject, { 385 wait: true, 386 data: { action: action }, 387 success: function( model) { 388 self.model.collection.remove( model ); 389 return self.clearRow(); 390 }, 391 error: function( model, response ) { 392 self.views.set( '#edit-group-member-' + model.get( 'id' ), new bp.Views.GroupMemberErrorInfo( { value: response.responseJSON.message } ).render() ); 393 394 // Make sure to reset request options. 395 model.resetRequestOptions(); 396 model.set( 'managingBan', false ); 397 } 398 } ); 399 } else if ( 'remove' === action ) { 400 this.model.set( 'removing', true ); 401 this.render(); 402 403 // Display user feedback. 404 this.views.set( '#edit-group-member-' + this.model.get( 'id' ), new bp.Views.GroupMemberUpdatingInfo( { value: action } ).render() ); 405 406 // Destroy the membership model. 407 this.model.destroy( { 408 wait: true, 409 data: {}, 410 success: function() { 411 return self.clearRow(); 412 }, 413 error: function( model, response ) { 414 self.views.set( '#edit-group-member-' + model.get( 'id' ), new bp.Views.GroupMemberErrorInfo( { value: response.responseJSON.message } ).render() ); 415 416 // Make sure to reset request options. 417 model.resetRequestOptions(); 418 model.set( 'removing', false ); 419 } 420 } ); 421 } 422 }, 423 424 editMemberRole: function( event ) { 425 var newRole = $( event.target ).val(), roleObject = this.getRoleObject( newRole ), 426 currentRole = this.model.get( 'role').id, roleAction = 'promote', self = this; 427 428 if ( newRole === this.model.get( 'role' ).id || ! roleObject ) { 429 return this.resetRow(); 430 } 431 432 this.views.set( '#edit-group-member-' + this.model.get( 'id' ), new bp.Views.GroupMemberUpdatingInfo().render() ); 433 434 if ( 'admin' === currentRole || ( 'mod' === currentRole && 'member' === newRole ) ) { 435 roleAction = 'demote'; 436 } 437 438 // Update Group member's role 439 this.model.save( roleObject, { 440 wait: true, 441 data: { 442 action: roleAction, 443 role: newRole 444 }, 445 success: function( model ) { 446 if ( self.model.collection.currentRole && newRole !== self.model.collection.currentRole ) { 447 self.model.collection.remove( model ); 448 return self.clearRow(); 449 } else { 450 return self.resetRow(); 451 } 452 }, 453 error: function( model, response ) { 454 self.views.set( '#edit-group-member-' + model.get( 'id' ), new bp.Views.GroupMemberErrorInfo( { value: response.responseJSON.message } ).render() ); 455 456 // Make sure to reset request options. 457 model.resetRequestOptions(); 458 model.set( 'editing', false ); 459 } 460 } ); 461 } 462 } ); 463 464 bp.Views.GroupMembersListHeader = bp.View.extend( { 465 tagName: 'thead', 466 template : bp.template( 'bp-manage-members-header' ) 467 } ); 468 469 bp.Views.GroupMembersListTable = bp.View.extend( { 470 tagName: 'tbody', 471 472 initialize: function() { 473 var preloaded = bpGroupManageMembersSettings.preloaded || {}, 474 models = []; 475 476 this.collection.on( 'reset', this.addListTableRows, this ); 477 478 if ( preloaded.body && preloaded.body.length > 0 ) { 479 _.each( preloaded.body, function( member ) { 480 models.push( new bp.Models.groupMember( member ) ); 481 } ); 482 483 this.collection.currentPage = 1; 484 if ( preloaded.headers && preloaded.headers[ 'X-WP-TotalPages' ] ) { 485 this.collection.totalPages = parseInt( preloaded.headers[ 'X-WP-TotalPages' ], 10 ); 486 } 487 488 if ( preloaded.headers && preloaded.headers[ 'X-WP-Total' ] ) { 489 this.collection.totalGroupMembers = parseInt( preloaded.headers[ 'X-WP-Total' ], 10 ); 490 } 491 492 this.collection.reset( models ); 493 } else { 494 this.collection.fetch( { 495 data: { 'exclude_admins': false }, 496 reset: true 497 } ); 498 } 499 }, 500 501 addListTableRows: function( collection ) { 502 if ( this.views._views ) { 503 var noMembersRow = _.findWhere( this.views._views[''] , { id: 'bp-no-group-members' } ); 504 505 if ( noMembersRow ) { 506 noMembersRow.remove(); 507 } 508 } 509 510 if ( ! collection.length ) { 511 this.views.add( new bp.Views.GroupMembersNoMatches( { id: 'bp-no-group-members' } ) ); 512 } else { 513 _.each( collection.models, function( member ) { 514 this.views.add( new bp.Views.GroupMembersListRow( { model: member } ) ); 515 }, this ); 516 } 517 } 518 } ); 519 520 bp.Views.GroupMembersUI = bp.View.extend( { 521 className: 'group-members', 522 523 initialize: function() { 524 var groupMembers = new bp.Collections.groupMembers(); 525 526 // Set filters. 527 this.views.set( '#group-roles-filter', [ 528 new bp.Views.GroupsMembersLabel( { attributes: { 'for': 'group-members-role-filter' } } ), 529 new bp.Views.GroupRolesDropDown( { id: 'group-members-role-filter', collection: groupMembers } ) 530 ] ); 531 532 // Set the search form. 533 this.views.set( '#group-members-search-form', new bp.Views.GroupMembersSearch( { id: 'group-members-search', collection: groupMembers } ) ); 534 535 // Set Paginate links. 536 this.views.set( '#group-members-pagination', new bp.Views.GroupsMembersPagination( { collection: groupMembers } ) ); 537 538 // Set Group members list header and body. 539 this.views.set( '#group-members-list-table', [ 540 new bp.Views.GroupMembersListHeader(), 541 new bp.Views.GroupMembersListTable( { collection: groupMembers } ) 542 ] ); 543 } 544 } ); 545 546 // Inject the UI to manage Group Members into the DOM. 547 bp.manageGroupMembersUI = new bp.Views.GroupMembersUI( { el:'#group-manage-members-ui' } ).render(); 548 549 } )( window.wp || {}, window.bp || {}, jQuery );
title
Description
Body
title
Description
Body
title
Description
Body
title
Body
Generated: Sat Sep 7 01:00:55 2024 | Cross-referenced by PHPXref 0.7.1 |