Laravel - 表单输入 - 多个选择一对多关系

时间:2014-07-08 09:09:28

标签: php forms laravel one-to-many

我正在构建的应用程序中的一个要求是表单输入,它为单个字段接收不同数量的项目。例如,我参加的运动是('足球','网球','门球')。

可以玩(可以说)有限数量的运动,所以这些项目应该从输入表格中的“下拉”类型列表中选择。

此表单的下游将是两个具有一对多关系的表。因此,从上面看,“user”表将有一行,而“user_sports”表将有三行。然后,这些将由用户表中的id字段链接。


我无法找到在文档中可以实现的功能(也许我不是在寻找正确的东西)。下面是我找到的最接近的,但仅用于从下拉列表中选择单个项目。

http://laravel.com/docs/html#drop-down-lists


是否有解决方法可以让我使用Laravel框架启动并运行此表单元素?

或者,还有其他方法可以实现这种功能,而不会损害用户体验吗?

7 个答案:

答案 0 :(得分:49)

我同意user3158900,我在使用方式上略有不同:

{{Form::label('sports', 'Sports')}}
{{Form::select('sports',$aSports,null,array('multiple'=>'multiple','name'=>'sports[]'))}}

但是,根据我的经验,select的第3个参数只是一个字符串,所以为了重新填充多选的数据,我必须做这样的事情:

<select multiple="multiple" name="sports[]" id="sports">
@foreach($aSports as $aKey => $aSport)
    @foreach($aItem->sports as $aItemKey => $aItemSport)
        <option value="{{$aKey}}" @if($aKey == $aItemKey)selected="selected"@endif>{{$aSport}}</option>
    @endforeach
@endforeach
</select>

答案 1 :(得分:41)

@SamMonk你的技术很棒。但是你可以使用laravel表单助手来完成。我有客户和狗的关系。

在您的控制器上

$dogs = Dog::lists('name', 'id');

在客户创建视图上,您​​可以使用。

{{ Form::label('dogs', 'Dogs') }}
{{ Form::select('dogs[]', $dogs, null, ['id' => 'dogs', 'multiple' => 'multiple']) }}

第三个参数接受一个井列表。如果您在模型上定义关系,则可以执行以下操作:

{{ Form::label('dogs', 'Dogs') }}
{{ Form::select('dogs[]', $dogs, $customer->dogs->lists('id'), ['id' => 'dogs', 'multiple' => 'multiple']) }}

Laravel 5.1更新

现在,lists方法返回一个Collection。 Upgrading To 5.1.0

{!! Form::label('dogs', 'Dogs') !!}
{!! Form::select('dogs[]', $dogs, $customer->dogs->lists('id')->all(), ['id' => 'dogs', 'multiple' => 'multiple']) !!}

答案 2 :(得分:6)

Laravel 4.2

@ SamMonk 提供了最佳选择,我按照他的示例构建了最后一段代码

<select class="chosen-select" multiple="multiple" name="places[]" id="places">
    @foreach($places as $place)
        <option value="{{$place->id}}" @foreach($job->places as $p) @if($place->id == $p->id)selected="selected"@endif @endforeach>{{$place->name}}</option>
    @endforeach
</select>

在我的项目中,我会有很多这样的表关系,所以我写了一个扩展来保持它的清洁。要加载它,请将它放在某些配置文件中,例如&#34; app / start / global.php&#34;。我创建了一个文件&#34; macros.php&#34;在&#34; app /&#34;目录并将其包含在global.php的EOF中

// app/start/global.php
require app_path().'/macros.php';

// macros.php
Form::macro("chosen", function($name, $defaults = array(), $selected = array(), $options = array()){

    // For empty Input::old($name) session, $selected is an empty string
    if(!$selected) $selected = array();

    $opts = array(
        'class' => 'chosen-select',
        'id' => $name,
        'name' => $name . '[]',
        'multiple' => true
    );
    $options = array_merge($opts, $options);
    $attributes = HTML::attributes($options);

    // need an empty array to send if all values are unselected
    $ret = '<input type="hidden" name="' . HTML::entities($name) . '[]">';
    $ret .= '<select ' . $attributes . '>';
    foreach($defaults as $def) {
        $ret .= '<option value="' . $def->id . '"';
        foreach($selected as $p) {
            // session array or passed stdClass obj
            $current = @$p->id ? $p->id: $p;
            if($def->id == $current) {
                $ret .= ' selected="selected"';
            }
        }
        $ret .= '>' . HTML::entities($def->name) . '</option>';
    }
    $ret .= '</select>';
    return $ret;
});

用法

没有预选项目的列表(创建视图)

{{ Form::chosen('places', $places, Input::old('places')) }}

预选(编辑视图)

{{ Form::chosen('places', $places, $job->places) }}

完整用法

{{ Form::chosen('places', $places, $job->places, ['multiple': false, 'title': 'I\'m a selectbox', 'class': 'bootstrap_is_mainstream']) }}

答案 3 :(得分:4)

多重选择实际上只是具有multiple属性的选择。考虑到这一点,它应该像......一样简单。

Form::select('sports[]', $sports, null, array('multiple'))

第一个参数只是名称,但使用[]对其进行后修复会在您使用Input::get('sports')时将其作为数组返回。

第二个参数是可选选项的数组。

第三个参数是您想要预先选择的选项数组。

第四个参数实际上是通过将multiple属性添加到实际的select元素来将其设置为多选下拉列表。

答案 4 :(得分:1)

My solution, it´s make with jquery-chosen and bootstrap, the id is for jquery chosen, tested and working, I had problems concatenating @foreach but now work with a double @foreach and double @if:

CustomerDao

this is the code por jquery chosen (the blade.php code doesn´t need this code to work)

  <div class="form-group">
    <label for="tagLabel">Tags: </label>
    <select multiple class="chosen-tag" id="tagLabel" name="tag_id[]" required>
      @foreach($tags as $id => $name)
        @if (is_array(Request::old('tag_id')))
                <option value="{{ $id }}" 
                @foreach (Request::old('tag_id') as $idold)
                  @if($idold==$id)
                    selected
                  @endif 
                @endforeach
                style="padding:5px;">{{ $name }}</option>
        @else
          <option value="{{ $id }}" style="padding:5px;">{{ $name }}</option>
        @endif
      @endforeach
    </select>
  </div>

答案 5 :(得分:0)

如果您需要将两个输出数组相互比较,但要使用第一个数组来填充选项,那么这可能是比最佳答案更好的方法。

当数组中有非数字或偏移量索引(键)时,这也很有用。

SELECT siph.SupplyPaymentID, 
si.SupplyInvoiceID, 
siph.DateOfPayment, 
si.TotalBill,
siph.BillPaidAmount, 
SUM(si.TotalBill-siph.BillPaidAmount) OVER(ORDER BY siph.DateOfPayment ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW) AS RemainingBillAmount 
FROM SupplyInvoicePaymentHistory siph 
INNER JOIN SupplyInvoice si ON siph.SupplyInvoiceID = si.SupplyInvoiceID  

答案 6 :(得分:0)

只要条件

<select name="category_type[]" id="category_type" class="select2 m-b-10 select2-multiple" style="width: 100%" multiple="multiple" data-placeholder="Choose" tooltip="Select Category Type">
 @foreach ($categoryTypes as $categoryType)
  <option value="{{ $categoryType->id }}"
    **@if(in_array($categoryType->id,
     request()->get('category_type')??[]))selected="selected"
    @endif**>
     {{ ucfirst($categoryType->title) }}</option>
     @endforeach
 </select>