Class: Command::List

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

Constant Summary collapse

ANNOTATION_COLOR_TIME_LIMIT =

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

6 * 60 * 60
NOVEL_TYPE_LABEL =

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

nil.to_i → 0 という互換性維持のため
%w(連載 連載 短編)
FILTERS_TYPE =
Hash[*%w(series 連載 ss 短編 frozen 凍結 nonfrozen 非凍結)]

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.



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
90
91
92
93
94
95
96
97
98
99
# File 'lib/command/list.rb', line 22

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

  ・現在管理している小説の一覧を表示します
  ・表示される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 -f "ss frozen"   # 凍結している短編だけ表示

# 小説家になろうの小説のみを表示
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

# リストをそのまま保存したい時(echoオプション)
narou l -e > list.txt

  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,
          "表示を絞るためのフィルター。スペース区切りで複数可\n" \
          "#{' '*25}#{filter_type_help}") { |filter|
    @options["filters"] = filter.downcase.split
  }
  @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
  }
  @opt.on("-e", "--echo", "パイプやリダイレクトでもそのまま出力する") {
    @options["echo"] = true
  }
end

Class Method Details

.oneline_helpObject



18
19
20
# File 'lib/command/list.rb', line 18

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

Instance Method Details

#execute(argv) ⇒ Object



223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
# File 'lib/command/list.rb', line 223

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.size
  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

#filter_type_helpObject



109
110
111
112
113
# File 'lib/command/list.rb', line 109

def filter_type_help
  FILTERS_TYPE.map { |filter, info|
    "#{filter}(#{info})"
  }.join(",")
end

#headerObject



136
137
138
139
140
141
142
143
144
# File 'lib/command/list.rb', line 136

def header
  [
     " ID ", " 更新日 ",
     @options["kind"] ? "種別" : nil,
     @options["author"] ? "作者名" : nil,
     @options["site"] ? "サイト名" : nil,
     "     タイトル"
  ].compact.join(" | ")
end

#output_list(novels) ⇒ Object



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
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
# File 'lib/command/list.rb', line 146

def output_list(novels)
  now = Time.now
  today = now.strftime("%y/%m/%d")
  filters = @options["filters"] || []
  selected_lines = {}
  novels.each do |novel|
    novel_type = novel["novel_type"].to_i
    id = novel["id"]
    frozen = Narou.novel_frozen?(id)

    unless filters.empty?
      next unless test_filter(filters, novel_type, frozen)
    end

    if @options["tags"]
      next unless valid_tags?(novel, @options["tags"])
    end
    disp_id = ((frozen ? "*" : "") + id.to_s).rjust(4)
    disp_id = disp_id.sub("*", "<bold><cyan>*</cyan></bold>") if frozen
    tags = novel["tags"] || []
    selected_lines[id] = [
      disp_id,
      novel["last_update"].strftime("%y/%m/%d").tap { |s|
        new_arrivals_date = novel["new_arrivals_date"]
        last_update = novel["last_update"]
        if new_arrivals_date && new_arrivals_date >= last_update \
           && new_arrivals_date + ANNOTATION_COLOR_TIME_LIMIT >= now
          # 新着表示色
          s.replace "<bold><magenta>#{s}</magenta></bold>"
        elsif last_update + ANNOTATION_COLOR_TIME_LIMIT >= now
          # 更新だけあった色
          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>" :
                       "") +
                       (tags.include?("end") ? " <bold><black>(完結)</black></bold>" : "") +
                       (tags.include?("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(" | ")
  end
  if @options["grep"]
    @options["grep"].each do |search_word|
      selected_lines.keep_if { |_, line|
        if search_word =~ /^-(.+)/
          # NOT検索
          !line.include?($1)
        else
          line.include?(search_word)
        end
      }
    end
  end
  if STDOUT.tty?
    puts header
    puts selected_lines.values.join("\n").termcolor
  else
    if @options["echo"]
      # pipeにそのまま出力するときはansicolorコードが邪魔なので削除
      puts header
      puts TermColorLight.strip_tag(selected_lines.values.join("\n"))
    else
      # pipeに接続するときはIDを渡す
      puts selected_lines.keys.join(" ")
    end
  end
end

#test_filter(filters, novel_type, frozen) ⇒ Object



115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
# File 'lib/command/list.rb', line 115

def test_filter(filters, novel_type, frozen)
  apply_sum = filters.inject(0) do |sum, item|
    apply = case item
            when "series"
              novel_type == 0 && novel_type == 1
            when "ss"
              novel_type == 2
            when "frozen"
              frozen
            when "nonfrozen"
              !frozen
            else
              error "不明なフィルターです(#{item})"
              warn "filters = #{filter_type_help}"
              exit Narou::EXIT_ERROR_CODE
            end
    sum + (apply ? 1 : 0)
  end
  apply_sum == filters.count
end

#valid_tags?(novel, tags) ⇒ Boolean

Returns:

  • (Boolean)


101
102
103
104
105
106
107
# File 'lib/command/list.rb', line 101

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