1

我对 Rails 还是很陌生,但我会说进展相当顺利。因此,为了练习,我正在开发一个简单的应用程序,用户可以在其中输入他们的体重,并且使用 Lazy Highcharts gem 通过 Highcharts 折线图向他们显示 30 天内的信息。我跟着 Ryan Bates 的 Railscast #223 开始。

我的问题:
好的,所以输入显示,除了在用户没有输入值的那一天,它在图表上显示为“0”(线下降到图表底部),而不是连接到任何一天的下一个点。不知道这一切是否有意义,所以这里有一个截图:

截屏

我找到了这个解决方案: Highcharts - Rails Array Includes Empty Date Entries

但是,当我实现在该页面上找到的第一个选项时(将 0 转换为 null):

(30.days.ago.to_date..Date.today).map { |date| wt = Weight.pounds_on(date).to_f; wt.zero? ? "null" : wt }

点出现了,但线没有出现,工具提示也没有......这让我认为有些东西正在破坏 js。不过,控制台中显然没有任何问题..?从这里我认为这可能是使用 Highchart 的 'connectNulls' 选项的问题,但这也不起作用。实施第二个选项(拒绝方法)时:

(30.days.ago.to_date..Date.today).map { |date| Weight.pounds_on(date).to_f}.reject(&:zero?)

它完全从图表中删除了所有为空的日期,这完全弄乱了结构,因为这些值应该根据它们的 created_at 日期显示。

所以回到第一方,这就是我剩下的(图表在没有输入的日子里绘制零)。

模型:

class Weight < ActiveRecord::Base
  attr_accessible :notes, :weight_input

  validates :weight_input, presence: true
  validates :user_id, presence: true

  belongs_to :user

  def self.pounds_on(date)
    where("date(created_at) = ?", date).pluck(:weight_input).last
  end
end

控制器:

def index
  @weights = current_user.weights.all

  @startdate = 30.days.ago.to_date
  @pounds = (30.days.ago.to_date..Date.today).map { |date| Weight.pounds_on(date).to_f }

  @h = LazyHighCharts::HighChart.new('graph') do |f|
    f.options[:title][:text] = " "
    f.options[:chart][:defaultSeriesType] = "area"
    f.options[:chart][:inverted] = false
    f.options[:chart][:zoomType] = 'x'
    f.options[:legend][:layout] = "horizontal"
    f.options[:legend][:borderWidth] = "0"
    f.series(:pointInterval => 1.day, :pointStart => @startdate, :name => 'Weight (lbs)', :color => "#2cc9c5", :data => @pounds )
    f.options[:xAxis] = {:minTickInterval => 1, :type => "datetime", :dateTimeLabelFormats => { day: "%b %e"}, :title => { :text => nil }, :labels => { :enabled => true } }
  end

  respond_to do |format|
    format.html # index.html.erb
    format.json { render json: @weights }
  end
end

有人对此有解决方案吗?我想我可能会做错这一切,所以非常感谢任何帮助。

4

1 回答 1

1

这是一个 HighCharts 特定的。 您需要将时间戳传递到您的数据数组中,而不是为数据集定义它。

对于每个数据点,我将 [time,value] 设置为一个元组。
[ "Date.UTC(#{date.year}, #{date.month}, #{date.day})" , Weight.pounds_on(date).to_f ]

您仍然需要以您喜欢的方式删除零,但数据将保持正确的日期值。

我认为你需要删除:pointInterval => 1.day

一般提示 您还应该考虑优化您对 pound_on 的查询,因为您正在对图表上的每个点进行数据库调用。执行一个Weight.where(:created_at => date_start..date_end ).group("Date(created_at)").sum(:weight_input) 将为您提供 created_at 日期数组以及每天总和的数组。

补充

改进的 SQL 查询

这依靠 sql 来做它最擅长的事情。首先,使用 where 将查询缩减为所需的记录(在本例中为过去 30 天)。选择您需要的字段(created_at 和 weight_input)。然后启动运行 sub_query 以按天对记录进行分组的内部联接,选择 created_at 的最大值。当它们匹配时,它会返回给定日期的最大(也就是最后输入的)重量输入。

@last_weight_per_day = Weight.where(:created_at => 30.days.ago.beginning_of_day..Time.now.end_of_day)
select("weights.created_at , weights.weight_input").
joins(" 
  inner join (  
      SELECT weights.weight_input, max(weights.created_at) as max_date 
      FROM weights 
      GROUP BY weights.weight_input , date(weights.created_at) 
  ) weights_dates on weights.created_at =  weights_dates.max_date
")

有了这个,你应该可以@last_weight_per_day这样。假设您已经在数据库中验证了它们,那么这不应该有 0 / nil 值。而且也应该很快。

  @pounds = @last_weight_per_day.map{|date| date.weight_input.to_f}
于 2013-05-05T16:10:19.207 回答