reactjs ToggleButtonGroup与类型无线电不起作用

时间:2018-06-01 13:07:55

标签: twitter-bootstrap reactjs

我正在尝试在reactjs es6中实现按钮组。现在我已将此示例https://react-bootstrap.github.io/components/button-group/与复选框一起使用,我正在尝试实现,但它似乎没有触发onchange事件。

    class ToggleButtonGroupControlled extends React.Component {
          constructor(props, context) {
            super(props, context);

            //this.handleChange = this.handleChange.bind(this);

            this.state = {
              value: 1
            };
          }

          handleChange(e) {
              console.log(e);
            this.setState({ value: e });
          }

          render() {
            return (
              <ToggleButtonGroup
                type="radio"
                    name="exemption_status_button"
                value={this.state.value}
                onChange={this.handleChange.bind(this)}
              defaultChecked={this.state.value}
              >
                <ToggleButton onChange={this.handleChange}value={1} className="badge2" data-count={this.props.statusButton.requested}>Requested</ToggleButton>
                <ToggleButton onChange={this.handleChange}value={3} className="badge2" data-count={this.props.statusButton.accepted}>Accepted</ToggleButton>
                <ToggleButton onChange={this.handleChange}value={6} className="badge2" data-count={this.props.statusButton.implemented}>Implemented</ToggleButton>
                <T

oggleButton onChange={this.handleChange}value={5} className="badge2" data-count={this.props.statusButton.cancelled}>Cancelled</ToggleButton>
            <ToggleButton onChange={this.handleChange}value={4} className="badge2" data-count={this.props.statusButton.rejected}>Rejected</ToggleButton>
          </ToggleButtonGroup>
        );
      }
    }

我按照以下方式工作。我不得不使用jquery来提取值并使用回传函数。

import _ from 'lodash';
import axios from 'axios';
import React, { Component } from 'react';
import { Checkbox } from 'react-bootstrap';
import { connect } from 'react-redux';
import {
    fetchSecurityExemptions,
    fetchSecurityExemptionsCount,
    fetchMarkets,
    fetchServiceDomains,
    fetchServicesList,
    fetchSecurityEngineerList,
    fetchBusinessPriorityList,
    updateSecurityExemption
} from '../../actions/security';
import { bindActionCreators } from 'redux';
import { Button, ButtonToolbar, Modal, Tooltip, OverlayTrigger, ToggleButton, ToggleButtonGroup } from 'react-bootstrap';
import FontAwesome from 'react-fontawesome';
import { BootstrapTable, TableHeaderColumn, InsertModalHeader, InsertModalFooter, SearchField } from 'react-bootstrap-table';
import 'react-bootstrap-table/dist/react-bootstrap-table-all.min.css';
import $ from 'jquery';
import ModalError from '../modalerror';
import ModalConfirm from '../modalconfirm';
import { Link } from 'react-router-dom';
import { isGenericName } from '../../scripts/validation';

class ToggleButtonGroupControlled extends Component {
      constructor(props, context) {
        super(props, context);

        this.handleChange = this.handleChange.bind(this);

        this.value = 1

      }

      handleChange(e) {
        this.value = $(e.target).children('input').val();
        this.passStatusButtonBack();
      }

      passStatusButtonBack(){
          this.props.passStatusButtonBack(this.value);
      }

      render() {
        return (
        <ButtonToolbar>
          <ToggleButtonGroup
            type="radio"
            name="exemption_status_button"
            value={this.value}
            onChange={this.handleChange}
            defaultChecked={this.value}
          >
            <ToggleButton onClick={this.handleChange}value={1} className="badge2" data-count={this.props.statusButton.requested}>Requested</ToggleButton>
            <ToggleButton onClick={this.handleChange}value={3} className="badge2" data-count={this.props.statusButton.accepted}>Accepted</ToggleButton>
            <ToggleButton onClick={this.handleChange}value={6} className="badge2" data-count={this.props.statusButton.implemented}>Implemented</ToggleButton>
            <ToggleButton onClick={this.handleChange}value={5} className="badge2" data-count={this.props.statusButton.cancelled}>Cancelled</ToggleButton>
            <ToggleButton onClick={this.handleChange}value={4} className="badge2" data-count={this.props.statusButton.rejected}>Rejected</ToggleButton>
          </ToggleButtonGroup>
        </ButtonToolbar>
        );
      }
    }


class SecurityExemptions extends Component {
    constructor(props) {
        super(props);
        this.securityExemptions = [];
        this.state = {
            viewport: {
                width: window.innerWidth,
                height: window.innerHeight
            },
            data: [],
            show: false,
            showError: false,
            errorMsg: '',
            errorTitle: '',
            confirm: {
                body: '',
                show: false,
                id: '',
                next: null
            },
            totalDataSize: 0,
            currentPage: 1,
            statusButton: {value: 1, requested: 0, accepted: 0, implemented: 0, cancelled:0, rejected: 0}
        };
        this.where = [];
        this.sort = 'id desc';
        this.meta = { title: '', description: 'Lists and manages security exemptions' };
        this.options = {
            noDataText: 'Loading...',
            defaultSortName: 'id',
            defaultSortOrder: 'desc',
            page: 1,
            paginationShowsTotal: true,
            sizePerPage: 50,
            sizePerPageList: [10, 25, 50, 100, 250],
            onPageChange: this.onPageChange.bind(this),
            onSizePerPageList: this.onSizePerPageList.bind(this),
            onFilterChange: this.onFilterChange.bind(this),
            onSortChange: this.onSortChange.bind(this),
            selectRowProp: {
                mode: 'checkbox',
                clickToSelect: true,
                onSelect: this.handleRowSelect,
                customComponent: this.customMultiSelect.bind(this)
            },
            cellEditProp: {
                mode: 'click',
                blurToSave: true,           
                afterDeleteRow: this.onAfterDeleteRow.bind(this),
                afterSaveCell: this.onAfterSaveCell.bind(this),
                nonEditableRows: function() {
                    return this.state.data.filter(row => row.exemption_status_id != 1 ).map(row => row.id);
                  }.bind(this)
            },
            handleConfirmDeleteRow: this.customConfirm.bind(this),
            btnGroup: this.createCustomButtonGroup.bind(this),
            deleteBtn: this.createCustomDeleteButton.bind(this),
            onSearchChange: this.onSearchChange.bind(this),
            searchField: this.createCustomSearchField            
        };

        this.updateWindowDimensions = this.updateWindowDimensions.bind(this);
        this.handleShowError = this.handleShowError.bind(this);
        this.fetchSecurityExemptions = this.fetchSecurityExemptions.bind(this);
        this.handlestatusButtonChange = this.handlestatusButtonChange.bind(this);
        this.runOnce = false;
        this.passStatusButtonBack = this.passStatusButtonBack.bind(this)

        this.markets = [];
        this.serviceDomain = [];
        this.servicesList = [];
        this.servicesListDisplay = [];
        this.securityEngineerList = [];
        this.businessPriority = []        
    }

    passStatusButtonBack(status){
        let statusButton = this.state.statusButton;
        statusButton.value = status;
        this.setState({ statusButton });
        this.where.exemption_status_id = status;
        this.fetchSecurityExemptions(this.options.page, this.options.sizePerPage, this.where, this.sort);
    }    


    passMetaBack = () => {
        this.props.passMetaBack(this.meta);
    };



    createCustomButtonGroup = props => {    
            return (
                <ButtonGroup className="my-custom-class" sizeClass="btn-group-md">
                    {props.insertBtn}
                    {props.deleteBtn}
                    <ExemptionStatusButtons self={this} />
                    <ToggleButtonGroupControlled statusButton={this.state.statusButton} passStatusButtonBack={this.passStatusButtonBack}/>
                </ButtonGroup>
            );

    };



    componentDidMount() {
        this.passMetaBack();
        this.updateWindowDimensions();
        window.addEventListener('resize', this.updateWindowDimensions);
    }

    componentWillUnmount() {
        window.removeEventListener('resize', this.updateWindowDimensions);
    }

    renderTable(tableData) {
        let insertRow = false;
        let deleteRow = false;

        return (
            <div className="container-fluid">
                <div className="row-fluid">
                    <BootstrapTable 
                        data={tableData} 
                        remote 
                        search 
                        fetchInfo={{ dataTotalSize: this.state.totalDataSize }} 
                        options={this.options} 
                        pagination 
                        striped 
                        hover 
                        tableHeaderClass="table-vf thead" 
                        tableContainerClass="securityExemptions" 
                        insertRow={insertRow} 
                        cellEdit={this.options.cellEditProp} 
                        deleteRow={deleteRow} 
                        selectRow={this.options.selectRowProp} 
                        maxHeight={this.state.viewport.height * 0.66} 
                        ref="table"
                    >
                        <TableHeaderColumn dataField="id" isKey={true} editable={false} hiddenOnInsert width={`60px`} dataFormat={this.exemptionsLink}>
                            ID
                        </TableHeaderColumn>
                        <TableHeaderColumn dataField="hits" editable={false} hiddenOnInsert width={`50px`}>
                            Hits
                        </TableHeaderColumn>
                        <TableHeaderColumn dataField="market" width={`150px`} formatExtraData={this.markets} editable={{ type: 'select', options: { values: this.markets } }}>
                            Market
                        </TableHeaderColumn>
                        <TableHeaderColumn dataField="service_domain" width={`150px`} formatExtraData={this.serviceDomain} editable={{ type: 'select', options: { values: this.serviceDomain } }}>
                            Service Domain
                        </TableHeaderColumn>
                        <TableHeaderColumn dataField="service_id" width={`200px`}  editable={false} dataFormat={this.serviceFormatter}>
                            Service / Project / Programme
                        </TableHeaderColumn>
                        <TableHeaderColumn dataField="demand_ref"  width={`70px`} editable={{ type: 'text', validator: this.isGenericName }}>
                            Demand ID
                        </TableHeaderColumn>
                        <TableHeaderColumn dataField="summary"  width={`200px`} editable={false}>
                            Summary
                        </TableHeaderColumn>
                        <TableHeaderColumn dataField="filename" width={`200px`} dataFormat={this.commsMatrixLink} editable={false}>
                            Commsmatrix
                        </TableHeaderColumn>
                        <TableHeaderColumn dataField="business_priority" width={`100px`} formatExtraData={this.businessPriority} editable={{ type: 'select', options: { values: this.businessPriority } }}>
                            Priority
                        </TableHeaderColumn>
                        <TableHeaderColumn dataField="expiry_date" width={`150px`} editable={true}>
                            Expiry
                        </TableHeaderColumn>
                        <TableHeaderColumn dataField="security_engineer" width={`200px`} dataFormat={this.htmlFormatter} formatExtraData={this.securityEngineerList} editable={{ type: 'select', options: { values: this.securityEngineerList } }}>
                            Security Engineer
                        </TableHeaderColumn>
                        <TableHeaderColumn dataField="ts_created" width={`130px`} editable={false} hiddenOnInsert>
                            Requested
                        </TableHeaderColumn>
                        <TableHeaderColumn dataField="created_by" width={`110px`} editable={false} hiddenOnInsert>
                            Requested By
                        </TableHeaderColumn>
                    </BootstrapTable>
                </div>
            </div>
        );
    }

    render() {
        const { securityExemptions } = this.props;

        if (!this.runOnce && this.props.isReady && this.securityExemptions.length == 0) {
            this.runOnce = true;

            this.where = {exemption_status_id : this.state.statusButton.value};
            this.initData();
        }

        let table = (
            <div>
                Loading...<i className="fa fa-spinner fa-spin" />
            </div>
        );

        if (this.state.show) {
            table = this.renderTable(this.state.data);
            this.moveElement();
        }

        return (
            <div className="container-fluid">
                <div className="row-fluid top-buffer">{table}</div>
                <ModalConfirm body={this.state.confirm.body} show={this.state.confirm.show} id={this.state.confirm.id} handleConfirmYes={this.handleConfirmYes} handleConfirmNo={this.handleConfirmNo} />
                <ModalError title={this.state.errorTitle} body={this.state.errorMsg} show={this.state.showError} handleCloseError={this.handleCloseError} />
            </div>
        );
    }
}

function mapStateToProps(state) {
    return {
        securityExemptions: state.securityExemptions,
        listsReducer: state.listsReducer
    };
}

function mapDispatchToProps(dispatch) {
    return bindActionCreators(
        {
            fetchSecurityExemptions,
            fetchSecurityExemptionsCount,
            fetchMarkets,
            fetchServiceDomains,
            fetchServicesList,
            fetchSecurityEngineerList,
            fetchBusinessPriorityList,
            updateSecurityExemption
        },
    dispatch
);

}

export default connect(mapStateToProps,mapDispatchToProps)(SecurityExemptions);

}

1 个答案:

答案 0 :(得分:0)

根据文档,onChange应该在ToggleButtonGroup组件上提供,而不是在ToggleButton组件上,

<ToggleButtonGroup
        type="checkbox"
        value={this.state.value}
        onChange={this.handleChange}
      >
        <ToggleButton value={1}>Checkbox 1 (pre-checked)</ToggleButton>
        <ToggleButton value={2}>Checkbox 2</ToggleButton>
        <ToggleButton value={3}>Checkbox 3 (pre-checked)</ToggleButton>
        <ToggleButton value={4} disabled>
          Checkbox 4 (disabled)
        </ToggleButton>
</ToggleButtonGroup>

编辑:如果这不起作用,您可以随时使用,

handleClick = (e)=>{
   console.log(e.target.value)
}

<ToggleButton onClick={this.handleClick}value={1} className="badge2" data-count={this.props.statusButton.requested}>Requested</ToggleButton>