Почему первый элемент всегда пуст в моем множественном выборе Rails, используя встроенный массив? - PullRequest
80 голосов
/ 19 января 2012

Я использую Rails 3.2.0.rc2 . У меня есть Model, в котором у меня есть статический Array, который я предлагаю через форму, так что пользователи могут выбрать подмножество Array и сохранить свой выбор в базе данных, сохраненной в одной столбец в Model. Я использовал serialize для столбца базы данных, в котором хранится Array, а Rails правильно преобразовывает выборки пользователей в Yaml (и обратно в массив при чтении этого столбца). Я использую множественный выбор формы для выбора.

Моя проблема в том, что, как у меня сейчас, все работает так, как я и ожидал, за исключением того, что массив поднаборов пользователя всегда имеет пустой первый элемент при отправке на сервер.

Это не имеет большого значения, и я мог бы написать код, чтобы обрезать это после свершившегося факта, но я чувствую, что просто делаю какую-то синтаксическую ошибку, так как мне не кажется, что Rails по умолчанию Поведение намеренно добавит этот пустой элемент без какой-либо причины. Должно быть, я что-то пропустил или забыл отключить какую-то настройку. Пожалуйста, помогите мне понять, чего мне не хватает (или укажите мне какую-нибудь хорошую документацию, которая описывает это более подробно, чем то, что я смог найти в промежутках).

Таблица базы данных MySQL 'models':

  • включает столбец с именем subset_array, который является текстовым полем

Модель класса включает в себя следующие настройки:

  • serialize :subset_array
  • ALL_POSSIBLE_VALUES = [value1, value2, value3, ...]

Форма для редактирования Модели включают следующую опцию ввода:

  • f.select :subset_array, Model::ALL_POSSIBLE_VALUES, {}, :multiple => true, :selected => @model.subset_array

PUT на сервер от клиента выглядит примерно так:

  • при условии, что выбраны только значение1 и значение3
  • "model" => { "subset_array" => ["", value1, value3] }

Обновление базы данных выглядит так:

  • UPDATE 'models' SET 'subset_array' = '--- \n- \"\"\n- value1\n- value3\n'

Как вы можете видеть, этот дополнительный пустой элемент в массиве отправляется и устанавливается в базу данных. Как мне избавиться от этого? Есть ли параметр, который мне не хватает в моем f.select вызове?

Большое спасибо, оценили:)

EDIT : это сгенерированный HTML-код из оператора f.select. Похоже, что создается скрытый ввод, который может быть причиной моей проблемы? Почему это там?

<input name="model[subset_array][]" type="hidden" value>
<select id="model_subset_array" multiple="multiple" name="model[subset_array][]" selected="selected">
    <option value="value1" selected="selected">Value1</option>
    <option value="value2">Value2</option>
    <option value="value3" selected="selected">Value3</option>
    <option...>...</option>
</select>

Ответы [ 9 ]

67 голосов
/ 20 января 2012

В Rails 4:

Вы сможете передать опцию :include_hidden.https://github.com/rails/rails/pull/5414/files

В качестве быстрого решения на данный момент: вы можете использовать прямо сейчас в вашей модели:

before_validation do |model|
  model.subset_array.reject!(&:blank?) if model.subset_array
end

Это просто удалит все пустые значения на уровне модели.

51 голосов
/ 20 января 2012

Скрытое поле является причиной проблемы. Но для этого есть веская причина: когда все значения отменены, вы все равно получаете параметр subset_array. Из документации Rails (возможно, вам придется прокрутить вправо, чтобы увидеть все это):

  # The HTML specification says when +multiple+ parameter passed to select and all options got deselected
  # web browsers do not send any value to server. Unfortunately this introduces a gotcha:
  # if an +User+ model has many +roles+ and have +role_ids+ accessor, and in the form that edits roles of the user
  # the user deselects all roles from +role_ids+ multiple select box, no +role_ids+ parameter is sent. So,
  # any mass-assignment idiom like
  #
  #   @user.update_attributes(params[:user])
  #
  # wouldn't update roles.
  #
  # To prevent this the helper generates an auxiliary hidden field before
  # every multiple select. The hidden field has the same name as multiple select and blank value.
  #
  # This way, the client either sends only the hidden field (representing
  # the deselected multiple select box), or both fields. Since the HTML specification
  # says key/value pairs have to be sent in the same order they appear in the
  # form, and parameters extraction gets the last occurrence of any repeated
  # key in the query string, that works for ordinary forms.

РЕДАКТИРОВАТЬ: Последний абзац предполагает, что вы не должны видеть пустой в случае, когда что-то выбрано, но я думаю, что это неправильно. Человек, который сделал этот коммит в Rails (см. https://github.com/rails/rails/commit/faba406fa15251cdc9588364d23c687a14ed6885), пытается сделать тот же трюк, который Rails использует для флажков (как упомянуто здесь: https://github.com/rails/rails/pull/1552),, но я не думаю, что это может работать для множественное поле выбора, потому что параметры, отправляемые через него, в этом случае образуют массив, поэтому никакие значения не игнорируются.

Так что я чувствую, что это ошибка.

11 голосов
/ 31 января 2012

Другое быстрое решение заключается в использовании этого фильтра контроллера:

def clean_select_multiple_params hash = params
  hash.each do |k, v|
    case v
    when Array then v.reject!(&:blank?)
    when Hash then clean_select_multiple_params(v)
    end
  end
end

Этот способ можно повторно использовать на контроллерах, не касаясь слоя модели.

8 голосов
/ 01 мая 2018

В Rails 4+ установлено: include_hidden для select_tag равно false

<%= form.grouped_collection_select :employee_id, Company.all, :employees, :name, :id, :name, { include_hidden: false }, { size: 6, multiple: true } %>
5 голосов
/ 03 апреля 2014

http://api.rubyonrails.org/classes/ActionView/Helpers/FormHelper.html#method-i-check_box

Попался

Спецификация HTML гласит, что снятые флажки или выборки не выполнены успешно, и, следовательно, веб-браузеры не отправляют их. К сожалению, это вводит Гоча: если модель Invoice имеет оплаченный флаг, и в форме, которая редактирует оплаченный счет, пользователь снимает флажок, не оплачен Параметр отправлен. Итак, любая идиома массового назначения вроде

@invoice.update (params [: invoice]) не будет обновлять флаг.

Чтобы предотвратить это, помощник создает вспомогательное скрытое поле перед Сам флажок. Скрытое поле имеет то же имя и его атрибуты имитируют непроверенный флажок.

Таким образом, клиент отправляет только скрытое поле (представляющее флажок снят) или оба поля. Поскольку HTML спецификация говорит, что пары ключ / значение должны быть отправлены в том же порядке они появляются в форме, а параметры извлечения получает последний появление любого повторяющегося ключа в строке запроса, который работает для обычные формы.

Чтобы удалить пустые значения:

  def myfield=(value)
    value.reject!(&:blank?)
    write_attribute(:myfield, value)
  end
3 голосов
/ 18 мая 2012

В контроллере:

arr = arr.delete_if { |x| x.empty? }
0 голосов
/ 14 октября 2014

Я исправил это с помощью params[:review][:staff_ids].delete("") в контроллере перед обновлением.

На мой взгляд:

= form_for @review do |f|
  = f.collection_select :staff_ids, @business.staff, :id, :full_name, {}, {multiple:true}
= f.submit 'Submit Review'

В моем контроллере:

class ReviewsController < ApplicationController
  def create
  ....
    params[:review][:staff_ids].delete("")
    @review.update_attribute(:staff_ids, params[:review][:staff_ids].join(","))
  ....
  end
end
0 голосов
/ 04 апреля 2013

Используйте jQuery:

$('select option:empty').remove(); 

Возможность удалить пустые опции из выпадающего списка.

0 голосов
/ 12 февраля 2013

Я заставляю это работать, написав это в части страницы Javascript:

$("#model_subset_array").val( <%= @model.subset_array %> );

Шахта больше похожа на следующую:

$("#modela_modelb_ids").val( <%= @modela.modelb_ids %> );

Не уверен, что в будущем это вызовет у меня головную боль, но теперь работает нормально.

...