7 #include "cmd-io/cmd-process-screen.h"
8 #include "cmd-visual/cmd-draw.h"
9 #include "core/asking-player.h"
10 #include "core/stuff-handler.h"
11 #include "core/visuals-reseter.h"
12 #include "game-option/special-options.h"
13 #include "io/files-util.h"
14 #include "io/input-key-acceptor.h"
15 #include "system/player-type-definition.h"
16 #include "system/redrawing-flags-updater.h"
17 #include "term/gameterm.h"
18 #include "term/screen-processor.h"
19 #include "term/term-color-types.h"
20 #include "util/angband-files.h"
21 #include "view/display-messages.h"
22 #include <string_view>
24 // Encode the screen colors
25 static char hack[17] = "dwsorgbuDWvyRGBU";
27 static concptr tags[4] = {
33 static concptr html_head[3] = {
34 "<html>\n<body text=\"#ffffff\" bgcolor=\"#000000\">\n",
38 static concptr html_foot[3] = {
45 * @brief 一時ファイルを読み込み、ファイルに書き出す
46 * @param fff ファイルへの参照ポインタ
47 * @param tempfff 一時ファイルへの参照ポインタ
49 * @todo io/ 以下に移したいところだが、このファイルの行数も大したことがないので一旦保留
51 static void read_temporary_file(FILE *fff, FILE *tmpfff, int num_tag)
53 bool is_first_line = true;
54 int next_tag = num_tag + 1;
56 while (!angband_fgets(tmpfff, buf, sizeof(buf))) {
58 if (strncmp(buf, tags[num_tag], strlen(tags[num_tag])) == 0) {
59 is_first_line = false;
65 if (strncmp(buf, tags[next_tag], strlen(tags[next_tag])) == 0) {
69 fprintf(fff, "%s\n", buf);
77 * @param fff 記念撮影ファイルへの参照ポインタ
79 static void screen_dump_one_line(int wid, int y, FILE *fff)
81 TERM_COLOR a = 0, old_a = 0;
83 for (TERM_LEN x = 0; x < wid - 1; x++) {
85 (void)(term_what(x, y, &a, &c));
101 c = (a == 0x09) ? '%' : '#';
107 if ((y == 0 && x == 0) || a != old_a) {
108 int rv = angband_color_table[a][1];
109 int gv = angband_color_table[a][2];
110 int bv = angband_color_table[a][3];
111 fprintf(fff, "%s<font color=\"#%02x%02x%02x\">",
112 ((y == 0 && x == 0) ? "" : "</font>"), rv, gv, bv);
117 fprintf(fff, "%s", cc);
119 fprintf(fff, "%c", c);
125 * @brief 記念撮影を行方向にスイープする
128 * @param fff 記念撮影ファイルへの参照ポインタ
130 static void screen_dump_lines(int wid, int hgt, FILE *fff)
132 for (TERM_LEN y = 0; y < hgt; y++) {
137 screen_dump_one_line(wid, y, fff);
142 * @brief ファイルへ書き込めない場合にエラーを表示する
143 * @param fff ダンプファイルへの参照ポインタ
145 * @return ファイルへ書き込めるならTRUE、書き込めないならFALSE
147 static bool check_screen_html_can_open(FILE *fff, char *filename, int message)
156 msg_format(_("ファイル %s を開けませんでした。", "Failed to open file %s."), filename);
162 * @brief HTMLヘッダを書き込む
163 * @param tmpfff 一時ファイルへの参照ポインタ
164 * @param fff 記念撮影ファイルへの参照ポインタ
166 static void write_html_header(FILE *tmpfff, FILE *fff)
169 read_temporary_file(fff, tmpfff, 0);
173 for (int i = 0; html_head[i]; i++) {
174 fputs(html_head[i], fff);
179 * @brief HTMLフッタを書き込む
180 * @param tmpfff 一時ファイルへの参照ポインタ
181 * @param fff 記念撮影ファイルへの参照ポインタ
183 static void write_html_footer(FILE *tmpfff, FILE *fff)
185 fprintf(fff, "</font>");
187 for (int i = 0; html_foot[i]; i++) {
188 fputs(html_foot[i], fff);
192 read_temporary_file(fff, tmpfff, 2);
193 angband_fclose(tmpfff);
199 void do_cmd_save_screen_html_aux(char *filename, int message)
202 term_get_size(&wid, &hgt);
203 auto *fff = angband_fopen(filename, FileOpenMode::WRITE);
204 if (!check_screen_html_can_open(fff, filename, message)) {
212 const auto &path = path_build(ANGBAND_DIR_USER, "htmldump.prf");
213 auto *tmpfff = angband_fopen(path, FileOpenMode::READ);
214 write_html_header(tmpfff, fff);
215 screen_dump_lines(wid, hgt, fff);
216 write_html_footer(tmpfff, fff);
219 msg_print(_("画面(記念撮影)をファイルに書き出しました。", "Screen dump saved."));
229 * @brief HTML方式で記念撮影する / Save a screen dump to a file
232 static void do_cmd_save_screen_html(void)
234 char tmp[256] = "screen.html";
235 if (!get_string(_("ファイル名: ", "File name: "), tmp, 80)) {
239 auto path = path_build(ANGBAND_DIR_USER, tmp);
240 auto filename = path.string();
242 do_cmd_save_screen_html_aux(filename.data(), 1);
246 * @brief 記念撮影の方式を問い合わせる
247 * @param html_dump HTMLダンプするか否か
248 * @return ダンプするならTRUE、キャンセルならFALSE
250 static bool ask_html_dump(bool *html_dump)
254 if (c == 'Y' || c == 'y') {
259 if (c == 'H' || c == 'h') {
273 * @brief ファイルへ書き込めない場合にエラーを表示する
274 * @param fff ダンプファイルへの参照ポインタ
275 * @return ファイルへ書き込めるならTRUE、書き込めないならFALSE
277 static bool check_screen_text_can_open(FILE *fff, const std::string_view filename)
283 msg_format(_("ファイル %s を開けませんでした。", "Failed to open file %s."), filename.data());
289 * @brief テキスト方式で記念撮影する
292 * @return 記念撮影に成功したらTRUE、ファイルが開けなかったらFALSE
293 * @todo どこかバグっていて、(恐らく初期化されていない)変な文字列まで出力される
295 static bool do_cmd_save_screen_text(int wid, int hgt)
299 const auto &path = path_build(ANGBAND_DIR_USER, "dump.txt");
300 auto *fff = angband_fopen(path, FileOpenMode::WRITE);
301 if (!check_screen_text_can_open(fff, path.string())) {
306 for (TERM_LEN y = 0; y < hgt; y++) {
309 for (x = 0; x < wid - 1; x++) {
310 (void)(term_what(x, y, &a, &c));
315 fprintf(fff, "%s\n", buf);
319 for (TERM_LEN y = 0; y < hgt; y++) {
322 for (x = 0; x < wid - 1; x++) {
323 (void)(term_what(x, y, &a, &c));
324 buf[x] = hack[a & 0x0F];
328 fprintf(fff, "%s\n", buf);
333 msg_print(_("画面(記念撮影)をファイルに書き出しました。", "Screen dump saved."));
340 * @brief 記念撮影のためにグラフィック使用をOFFにする
341 * @param player_ptr プレイヤーへの参照ポインタ
342 * @return 記念撮影直前のグラフィックオプション
344 static bool update_use_graphics(PlayerType *player_ptr)
350 use_graphics = false;
351 reset_visuals(player_ptr);
353 MainWindowRedrawingFlag::WIPE,
354 MainWindowRedrawingFlag::BASIC,
355 MainWindowRedrawingFlag::EXTRA,
356 MainWindowRedrawingFlag::MAP,
357 MainWindowRedrawingFlag::EQUIPPY,
359 RedrawingFlagsUpdater::get_instance().set_flags(flags);
360 handle_stuff(player_ptr);
365 * Save a screen dump to a file
366 * @param player_ptr プレイヤーへの参照ポインタ
368 void do_cmd_save_screen(PlayerType *player_ptr)
370 prt(_("記念撮影しますか? [(y)es/(h)tml/(n)o] ", "Save screen dump? [(y)es/(h)tml/(n)o] "), 0, 0);
372 if (!ask_html_dump(&html_dump)) {
377 term_get_size(&wid, &hgt);
379 bool old_use_graphics = update_use_graphics(player_ptr);
382 do_cmd_save_screen_html();
383 do_cmd_redraw(player_ptr);
384 } else if (!do_cmd_save_screen_text(wid, hgt)) {
388 if (old_use_graphics) {
393 reset_visuals(player_ptr);
395 MainWindowRedrawingFlag::WIPE,
396 MainWindowRedrawingFlag::BASIC,
397 MainWindowRedrawingFlag::EXTRA,
398 MainWindowRedrawingFlag::MAP,
399 MainWindowRedrawingFlag::EQUIPPY,
401 RedrawingFlagsUpdater::get_instance().set_flags(flags);
402 handle_stuff(player_ptr);
406 * @brief 白文字だけ画面に描画する
408 * @param fff 記念撮影ファイルへの参照ポインタ
412 * @return ファイルが読み込めなくなったらFALSEで抜ける
414 static bool draw_white_characters(FILE *fff, int wid, int hgt)
417 for (TERM_LEN y = 0; okay; y++) {
419 if (!fgets(buf, sizeof(buf), fff)) {
423 if (buf[0] == '\n' || buf[0] == '\0') {
430 for (TERM_LEN x = 0; x < wid - 1; x++) {
431 if (buf[x] == '\n' || buf[x] == '\0') {
435 term_draw(x, y, TERM_WHITE, buf[x]);
443 * @brief 白以外の文字を画面に描画する
444 * @param fff 記念撮影ファイルへの参照ポインタ
447 * @param 白文字が途中で読み込めなくなっていたらTRUE
450 static void draw_colored_characters(FILE *fff, int wid, int hgt, bool okay)
452 TERM_COLOR a = TERM_DARK;
454 for (TERM_LEN y = 0; okay; y++) {
456 if (!fgets(buf, sizeof(buf), fff)) {
460 if (buf[0] == '\n' || buf[0] == '\0') {
467 for (TERM_LEN x = 0; x < wid - 1; x++) {
468 if (buf[x] == '\n' || buf[x] == '\0') {
472 (void)(term_what(x, y, &a, &c));
473 for (int i = 0; i < 16; i++) {
474 if (hack[i] == buf[x]) {
479 term_draw(x, y, a, c);
485 * @brief Load a screen dump from a file
488 void do_cmd_load_screen(void)
491 term_get_size(&wid, &hgt);
492 const auto path = path_build(ANGBAND_DIR_USER, "dump.txt");
493 auto *fff = angband_fopen(path, FileOpenMode::READ);
495 const auto filename = path.string();
496 msg_format(_("%s を開くことができませんでした。", "Failed to open %s."), filename.data());
503 bool okay = draw_white_characters(fff, wid, hgt);
504 draw_colored_characters(fff, wid, hgt, okay);
507 prt(_("ファイルに書き出された画面(記念撮影)をロードしました。", "Screen dump loaded."), 0, 0);