lib/command/list.rb in narou-1.5.11 vs lib/command/list.rb in narou-1.6.0

- old
+ new

@@ -11,26 +11,47 @@ # MEMO: 0 は昔の小説を凍結したままな場合、novel_type が設定されていないので、 # nil.to_i → 0 という互換性維持のため NOVEL_TYPE_LABEL = ["連載", "連載", "短編"] + def self.oneline_help + "現在管理している小説の一覧を表示します" + end + def initialize - super("[<number>] [options]") - @opt.separator <<-EOS + super("[<limit>] [options]") + @opt.separator <<-EOS.termcolor - ・現在管理している小説の一覧を表示します。 - ・表示されるIDは各コマンドで指定することで小説名等を入力する手間を省けます。 + ・現在管理している小説の一覧を表示します + ・表示されるIDは各コマンドで指定することで小説名等を入力する手間を省けます ・個数を与えることで、最大表示数を制限できます(デフォルトは全て表示) ・narou listのデフォルト動作を narou s default_arg.list= で設定すると便利です + ・パイプで他のnarouコマンドに繋ぐとID入力の代わりにできます - Example: + 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 } @@ -38,59 +59,123 @@ @options["reverse"] = true } @opt.on("-u", "--url", "小説の掲載ページも表示する") { @options["url"] = true } - @opt.on("-t", "--type", "小説の種別(短編/連載)も表示する") { - @options["type"] = 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 + 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 + def output_list(novels) now = Time.now today = now.strftime("%y/%m/%d") filter = @options["filter"] - header = [" ID ", " 更新日 ", @options["type"] ? "種別" : nil, " タイトル"].compact - puts header.join(" | ") + 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("*", "<cyan>*</cyan>").termcolor if frozen - flags = novel["flags"] || {} - puts [ + 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>".termcolor + s.replace "<bold><magenta>#{s}</magenta></bold>" elsif s == today # 更新だけあった色 - s.replace "<bold><green>#{s}</green></bold>".termcolor + s.replace "<bold><green>#{s}</green></bold>" end }, - @options["type"] ? NOVEL_TYPE_LABEL[novel_type] : nil, - novel["title"] + (!@options["type"] && novel_type == 2 ? - " <gray>(#{NOVEL_TYPE_LABEL[novel_type]})</gray>".termcolor : + @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"] ? " <gray>(完結)</gray>".termcolor : "") + - (flags["delete"] ? " <gray>(削除)</gray>".termcolor : ""), - @options["url"] ? novel["toc_url"] : nil - ].compact.join(" | ") + (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 def execute(argv) super database_values = Database.instance.get_object.values @@ -103,12 +188,8 @@ database_values = Database.instance.sort_by_last_update end database_values.reverse! if @options["reverse"] novels = database_values[0, num] output_list(novels) - end - - def oneline_help - "現在管理している小説の一覧を表示します" end end end