14

デフォルトのRailsの日付、時刻、および日時フィールドを使用して、カピバラで日付セレクターを構築しようとしています。メソッドを使用しwithinてフィールドの選択ボックスを見つけていますが、xPath を使用して正しいボックスを見つけると、within範囲を離れて要素のページで最初に出現するものを見つけます。

これが私が使用しているコードです。私がテストしているページには2つの日時フィールドがありますが、このエラーのために最初のものしか変更できません。現時点では、datetime フィールドをラップする id を持つ div コンテナーがありますが、ラベルで検索するようにコードを切り替える予定です。

module  Marketron
  module DateTime

    def select_date(field, options = {})
      date_parse = Date.parse(options[:with])

      year = date_parse.year.to_s
      month = date_parse.strftime('%B')
      day = date_parse.day.to_s

      within("div##{field}") do
        find(:xpath, "//select[contains(@id, \"_#{FIELDS[:year]}\")]").select(year)
        find(:xpath, "//select[contains(@id, \"_#{FIELDS[:month]}\")]").select(month)
        find(:xpath, "//select[contains(@id, \"_#{FIELDS[:day]}\")]").select(day)
      end

    end

    def select_time(field, options = {})
      require "time"
      time_parse = Time.parse(options[:with])

      hour = time_parse.hour.to_s.rjust(2, '0')
      minute = time_parse.min.to_s.rjust(2, '0')

      within("div##{field}") do
        find(:xpath, "//select[contains(@id, \"_#{FIELDS[:hour]}\")]").find(:xpath, "option[contains(@value, '#{hour}')]").select_option
        find(:xpath, "//select[contains(@id, \"_#{FIELDS[:minute]}\")]").find(:xpath, "option[contains(@value, '#{minute}')]").select_option

      end

    end

    def select_datetime(field, options = {})
      select_date(field, options)
      select_time(field, options)
    end

    private

      FIELDS = {year: "1i", month: "2i", day: "3i", hour: "4i", minute: "5i"}

  end
end

World(Marketron::DateTime)
4

1 に答える 1

36

先頭に a を追加して、現在のノードから開始することを xpath で指定する必要があります.

find(:xpath, ".//select[contains(@id, \"_#{FIELDS[:year]}\")]")

例:

この HTML ページをテストしました (ページを単純化しすぎないことを願っています)。

<html>
    <div id='div1'>
        <span class='container'>    
            <span id='field_01'>field 1</span>
        </span>
    </div>
    <div id='div2'>
        <span class='container'>
            <span id='field_02'>field 2</span>
        </span>
    </div>  
</html>

within メソッドを使用すると、これを行うと問題を確認できます。

within("div#div1"){ puts find(:xpath, "//span[contains(@id, \"field\")]").text }
#=> field 1
within("div#div2"){ puts find(:xpath, "//span[contains(@id, \"field\")]").text }
#=> field 1

しかし、現在のノード内を参照するように xpath を指定すると (つまり、 を使用して.)、必要な結果が得られます。

within("div#div1"){ puts find(:xpath, ".//span[contains(@id, \"field\")]").text }
#=> field 1
within("div#div2"){ puts find(:xpath, ".//span[contains(@id, \"field\")]").text }
#=> field 2
于 2012-04-21T12:22:56.243 に答える