Class: Command::List

Inherits:
CommandBase show all
Defined in:
lib/command/list.rb

Constant Summary collapse

NEW_ARRIVALS_LIMIT =

更新してから何秒までを新着色にするか

6 * 60 * 60
NOVEL_TYPE_LABEL =

MEMO: 0 は昔の小説を凍結したままな場合、novel_type が設定されていないので、

nil.to_i → 0 という互換性維持のため
["連載", "連載", "短編"]

Class Method Summary collapse

Instance Method Summary collapse

Methods inherited from CommandBase

execute!, #force_change_settings_function, help, #hook_call, #load_local_settings, #tagname_to_ids

Constructor Details

#initializeList

Returns a new instance of List.



20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
# File 'lib/command/list.rb', line 20

def initialize
  super("[<limit>] [options]")
  @opt.separator <<-EOS.termcolor

  ・現在管理している小説の一覧を表示します
  ・表示されるIDは各コマンドで指定することで小説名等を入力する手間を省けます
  ・個数を与えることで、最大表示数を制限できます(デフォルトは全て表示)
  ・narou listのデフォルト動作を narou s default_arg.list= で設定すると便利です
  ・パイプで他のnarouコマンドに繋ぐとID入力の代わりにできます

  Examples:
narou list             # IDの小さい順に全て表示
narou list 10 -r       # IDの大きい順に10件表示
narou list 5 -l        # 最近更新のあった5件表示
narou list 10 -rl      # 古い順に10件表示
narou list -f ss       # 短編小説だけ表示

# 小説家になろうの小説のみを表示
narou list --site --grep 小説家になろう
narou l -sg 小説家になろう    # 上記と同じ意味
# 作者“紫炎”を含む小説を表示
narou list --author --grep 紫炎
narou l -ag 紫炎              # 上記と同じ意味
# “紫炎”と“なろう”を含む小説を表示(AND検索)
narou l -asg "紫炎 なろう"
# “なろう”を含まない小説を表示(NOT検索)
narou l -sg "-なろう"

# ハーメルンを含む小説にhamelnタグを付ける
narou l -sg ハーメルン | narou t -a hameln
# 短編を全て凍結する
narou l -f ss | narou freeze --on

  Options:
  EOS
  @opt.on("-l", "--latest", "最近更新のあった順に小説を表示する") {
    @options["latest"] = true
  }
  @opt.on("-r", "--reverse", "逆順に表示する") {
    @options["reverse"] = true
  }
  @opt.on("-u", "--url", "小説の掲載ページも表示する") {
    @options["url"] = true
  }
  @opt.on("-k", "--kind", "小説の種別(短編/連載)も表示する") {
    @options["kind"] = true
  }
  @opt.on("-s", "--site", "掲載小説サイト名も表示する") {
    @options["site"] = true
  }
  @opt.on("-a", "--author", "作者名も表示する") {
    @options["author"] = true
  }
  @opt.on("-f", "--filter VAL", String,
          "表示を絞るためのフィルターの種類(連載:series, 短編:ss)") { |filter|
    @options["filter"] = filter
  }
  @opt.on("-g", "--grep VAL", String,
          "指定された文字列でリストを検索する") { |search|
    @options["grep"] = search.split
  }
  @opt.on("-t", "--tag [TAGS]", String,
          "タグも表示。引数を指定した場合そのタグを含む小説を表示") { |tags|
    if tags
      @options["tags"] = tags.split
    else
      @options["all-tags"] = true
    end
  }
end

Class Method Details

.oneline_helpObject



16
17
18
# File 'lib/command/list.rb', line 16

def self.oneline_help
  "現在管理している小説の一覧を表示します"
end

Instance Method Details

#execute(argv) ⇒ Object



179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
# File 'lib/command/list.rb', line 179

def execute(argv)
  super
  database_values = Database.instance.get_object.values
  if !argv.empty? && argv.first =~ /^\d+$/
    num = argv.first.to_i
  else
    num = database_values.count
  end
  if @options["latest"]
    database_values = Database.instance.sort_by_last_update
  end
  database_values.reverse! if @options["reverse"]
  novels = database_values[0, num]
  output_list(novels)
end

#output_list(novels) ⇒ Object



99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
# File 'lib/command/list.rb', line 99

def output_list(novels)
  now = Time.now
  today = now.strftime("%y/%m/%d")
  filter = @options["filter"]
  if STDOUT.tty?
    header = [" ID ", " 更新日 ",
              @options["kind"] ? "種別" : nil,
              @options["author"] ? "作者名" : nil,
              @options["site"] ? "サイト名" : nil,
              "     タイトル"].compact
    puts header.join(" | ")
  end
  selected_lines = {}
  novels.each do |novel|
    novel_type = novel["novel_type"].to_i
    if filter
      if filter == "series" && novel_type != 0 && novel_type != 1
        next
      elsif filter == "ss" && novel_type != 2
        next
      end
    end
    if @options["tags"]
      next unless valid_tags?(novel, @options["tags"])
    end
    id = novel["id"]
    frozen = Narou.novel_frozen?(id)
    disp_id = ((frozen ? "*" : "") + id.to_s).rjust(4)
    disp_id = disp_id.sub("*", "<bold><cyan>*</cyan></bold>").termcolor if frozen
    flags = novel["flags"] || {}   # flagコマンドは1.6.0から非推奨
    tags = novel["tags"] || []
    flags["end"] ||= tags.include?("end")
    flags["404"] ||= tags.include?("404")
    selected_lines[id] = [
      disp_id,
      novel["last_update"].strftime("%y/%m/%d").tap { |s|
        if novel["new_arrivals_date"] && novel["new_arrivals_date"] + NEW_ARRIVALS_LIMIT >= now
          # 新着表示色
          s.replace "<bold><magenta>#{s}</magenta></bold>"
        elsif s == today
          # 更新だけあった色
          s.replace "<bold><green>#{s}</green></bold>"
        end
      },
      @options["kind"] ? NOVEL_TYPE_LABEL[novel_type] : nil,
      @options["author"] ? novel["author"].escape : nil,
      @options["site"] ? novel["sitename"].escape : nil,
      novel["title"].escape + (!@options["kind"] && novel_type == 2 ?
                       "  <bold><black>(#{NOVEL_TYPE_LABEL[novel_type]})</black></bold>" :
                       "") +
                       (flags["end"] ? " <bold><black>(完結)</black></bold>" : "") +
                       (flags["404"] ? " <bold><black>(削除)</black></bold>" : ""),
      @options["url"] ? novel["toc_url"].escape : nil,
      @options["tags"] || @options["all-tags"] ?
          tags.empty? ? nil : tags.map{ |tag|
            color = Tag.get_color(tag)
            "<bold><#{color}>#{tag.escape}</#{color}></bold>"
          }.join(",") : nil,
    ].compact.join(" | ").termcolor
  end
  if @options["grep"]
    @options["grep"].each do |search_word|
      selected_lines.keep_if { |id, line|
        if search_word =~ /^-(.+)/
          # NOT検索
          not line.include?($1)
        else
          line.include?(search_word)
        end
      }
    end
  end
  if STDOUT.tty?
    puts selected_lines.values
  else
    # pipeに接続するときはIDを渡す
    puts selected_lines.keys.join(" ")
  end
end

#valid_tags?(novel, tags) ⇒ Boolean

Returns:

  • (Boolean)


91
92
93
94
95
96
97
# File 'lib/command/list.rb', line 91

def valid_tags?(novel, tags)
  novel_tags = novel["tags"] or return false
  tags.each do |tag|
    return false unless novel_tags.include?(tag)
  end
  true
end