Class: Command::List

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

Defined Under Namespace

Classes: NovelDecorator

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 非凍結)]

Instance Attribute Summary collapse

Attributes inherited from CommandBase

#stream_io

Class Method Summary collapse

Instance Method Summary collapse

Methods inherited from CommandBase

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

Constructor Details

#initializeList

rubocop:disable Metrics/MethodLength



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
100
101
102
103
104
105
106
107
108
# File 'lib/command/list.rb', line 28

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("--gl", "更新日ではなく最新話掲載日を使用する") {
    @options["general-lastup"] = 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

Instance Attribute Details

#optionsObject (readonly)

Returns the value of attribute options.



21
22
23
# File 'lib/command/list.rb', line 21

def options
  @options
end

Class Method Details

.oneline_helpObject



23
24
25
# File 'lib/command/list.rb', line 23

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

Instance Method Details

#decorate_lines(novels) ⇒ Object



169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
# File 'lib/command/list.rb', line 169

def decorate_lines(novels)
  filters = @options["filters"] || []
  selected_lines = {}
  novels.each do |novel|
    novel_decorator = NovelDecorator.new(novel, self)
    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
    selected_lines[id] = novel_decorator.decorate_line
  end
  grep(selected_lines)
end

#execute(argv) ⇒ Object



219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
# File 'lib/command/list.rb', line 219

def execute(argv)
  disable_logging
  super
  database_values = Database.instance.get_object.values
  limit =
    if !argv.empty? && argv.first =~ /^\d+$/
      argv.first.to_i
    else
      database_values.size
    end
  if @options["latest"]
    database_values = Database.instance.sort_by(view_date_type)
  end
  database_values.reverse! if @options["reverse"]
  output_list(database_values, limit)
end

#filter_type_helpObject



118
119
120
121
122
# File 'lib/command/list.rb', line 118

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

#grep(selected_lines) ⇒ Object



190
191
192
193
194
195
196
197
198
199
200
201
202
203
# File 'lib/command/list.rb', line 190

def grep(selected_lines)
  return selected_lines unless @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
  selected_lines
end

#headerObject



145
146
147
148
149
150
151
152
153
154
# File 'lib/command/list.rb', line 145

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

#nowObject



160
161
162
# File 'lib/command/list.rb', line 160

def now
  @now ||= Time.now
end

#output_lines(took_lines) ⇒ Object



205
206
207
208
209
210
211
212
213
214
215
216
217
# File 'lib/command/list.rb', line 205

def output_lines(took_lines)
  if STDOUT.tty?
    stream_io.puts header
    stream_io.puts took_lines.values.join("\n").termcolor
  elsif @options["echo"]
    # pipeにそのまま出力するときはansicolorコードが邪魔なので削除
    stream_io.puts header
    stream_io.puts TermColorLight.strip_tag(took_lines.values.join("\n"))
  else
    # pipeに接続するときはIDを渡す
    stream_io.puts took_lines.keys.join(" ")
  end
end

#output_list(novels, limit) ⇒ Object



164
165
166
167
# File 'lib/command/list.rb', line 164

def output_list(novels, limit)
  took_lines = Hash[decorate_lines(novels).take(limit)]
  output_lines(took_lines)
end

#test_filter(filters, novel_type, frozen) ⇒ Object



124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
# File 'lib/command/list.rb', line 124

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
              stream_io.error "不明なフィルターです(#{item})"
              stream_io.puts "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)


110
111
112
113
114
115
116
# File 'lib/command/list.rb', line 110

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

#view_date_typeObject



156
157
158
# File 'lib/command/list.rb', line 156

def view_date_type
  @_view_data_type ||= @options["general-lastup"] ? "general_lastup" : "last_update"
end