local config_module = require('cp.config') local snippets = require('cp.snippets') local M = {} local config = {} local function log(msg, level) vim.notify(('[cp.nvim]: %s'):format(msg), level or vim.log.levels.INFO) end local function clearcol() vim.api.nvim_set_option_value('number', false, { scope = 'local' }) vim.api.nvim_set_option_value('relativenumber', false, { scope = 'local' }) vim.api.nvim_set_option_value('statuscolumn', '', { scope = 'local' }) vim.api.nvim_set_option_value('signcolumn', 'no', { scope = 'local' }) vim.api.nvim_set_option_value('equalalways', false, { scope = 'global' }) end local function get_plugin_path() local plugin_path = debug.getinfo(1, 'S').source:sub(2) return vim.fn.fnamemodify(plugin_path, ':h:h:h') end local function setup_python_env() local plugin_path = get_plugin_path() local venv_dir = plugin_path .. '/.venv' if vim.fn.executable('uv') == 0 then log( 'uv is not installed. Install it to enable problem scraping: https://docs.astral.sh/uv/', vim.log.levels.WARN ) return false end if vim.fn.isdirectory(venv_dir) == 0 then log('setting up Python environment for scrapers...') local result = vim.fn.system( ('cd %s && uv sync'):format(vim.fn.shellescape(plugin_path)) ) if vim.v.shell_error ~= 0 then log( 'failed to setup Python environment: ' .. result, vim.log.levels.ERROR ) return false end log('python environment setup complete') end return true end local competition_types = { 'atcoder', 'codeforces', 'cses', 'icpc' } local function setup_contest(contest_type) if not vim.tbl_contains(competition_types, contest_type) then log( ('unknown contest type. Available: [%s]'):format( table.concat(competition_types, ', ') ), vim.log.levels.ERROR ) return end vim.g.cp_contest = contest_type vim.fn.system(('cp -fr %s/* .'):format(config.template_dir)) vim.fn.system( ('make setup VERSION=%s'):format( config.contests[contest_type].cpp_version ) ) log(('set up %s contest environment'):format(contest_type)) end local function setup_problem(problem_id, problem_letter) if not vim.g.cp_contest then log( 'no contest mode set. run :CP first', vim.log.levels.ERROR ) return end if vim.g.cp_diff_mode then vim.cmd.diffoff() if vim.g.cp_saved_session then vim.fn.delete(vim.g.cp_saved_session) vim.g.cp_saved_session = nil end if vim.g.cp_temp_output then vim.fn.delete(vim.g.cp_temp_output) vim.g.cp_temp_output = nil end vim.g.cp_diff_mode = false end vim.cmd.only() local filename, full_problem_id if (vim.g.cp_contest == 'atcoder' or vim.g.cp_contest == 'codeforces') and problem_letter then full_problem_id = problem_id .. problem_letter filename = full_problem_id .. '.cc' vim.fn.system( ('make scrape %s %s %s'):format( vim.g.cp_contest, problem_id, problem_letter ) ) else full_problem_id = problem_id filename = problem_id .. '.cc' vim.fn.system( ('make scrape %s %s'):format(vim.g.cp_contest, problem_id) ) end vim.cmd.e(filename) if vim.api.nvim_buf_get_lines(0, 0, -1, true)[1] == '' then vim.api.nvim_input(('i%s'):format(vim.g.cp_contest)) end vim.api.nvim_set_option_value('winbar', '', { scope = 'local' }) vim.api.nvim_set_option_value('foldlevel', 0, { scope = 'local' }) vim.api.nvim_set_option_value('foldmethod', 'marker', { scope = 'local' }) vim.api.nvim_set_option_value('foldmarker', '{{{,}}}', { scope = 'local' }) vim.api.nvim_set_option_value('foldtext', '', { scope = 'local' }) vim.diagnostic.enable(false) local base_fp = vim.fn.fnamemodify(filename, ':p:h') local input = ('%s/io/%s.in'):format(base_fp, full_problem_id) local output = ('%s/io/%s.out'):format(base_fp, full_problem_id) vim.cmd.vsplit(output) vim.cmd.w() clearcol() vim.cmd(('vertical resize %d'):format(math.floor(vim.o.columns * 0.3))) vim.cmd.split(input) vim.cmd.w() clearcol() vim.cmd.wincmd('h') log(('switched to problem %s'):format(full_problem_id)) end local function get_current_problem() local filename = vim.fn.expand('%:t:r') if filename == '' then log('no file open', vim.log.levels.ERROR) return nil end return filename end local function run_problem() local problem_id = get_current_problem() if not problem_id then return end local has_lsp, lsp = pcall(require, 'lsp') if has_lsp and lsp.lsp_format then lsp.lsp_format({ async = true }) end vim.system({ 'make', 'run', vim.fn.expand('%:t') }, {}, function() vim.schedule(function() vim.cmd.checktime() end) end) end local function debug_problem() local problem_id = get_current_problem() if not problem_id then return end local has_lsp, lsp = pcall(require, 'lsp') if has_lsp and lsp.lsp_format then lsp.lsp_format({ async = true }) end vim.system({ 'make', 'debug', vim.fn.expand('%:t') }, {}, function() vim.schedule(function() vim.cmd.checktime() end) end) end local function diff_problem() if vim.g.cp_diff_mode then vim.cmd.diffoff() if vim.g.cp_saved_session then vim.cmd(('silent! source %s'):format(vim.g.cp_saved_session)) vim.fn.delete(vim.g.cp_saved_session) vim.g.cp_saved_session = nil end vim.g.cp_diff_mode = false log('exited diff mode') else local problem_id = get_current_problem() if not problem_id then return end local base_fp = vim.fn.getcwd() local output = ('%s/io/%s.out'):format(base_fp, problem_id) local expected = ('%s/io/%s.expected'):format(base_fp, problem_id) local input = ('%s/io/%s.in'):format(base_fp, problem_id) if vim.fn.filereadable(expected) == 0 then log( ('No expected output file found: %s'):format(expected), vim.log.levels.ERROR ) return end local temp_output = vim.fn.tempname() vim.fn.system( ("awk '/^\\[[^]]*\\]:/ {exit} {print}' %s > %s"):format( vim.fn.shellescape(output), temp_output ) ) local session_file = vim.fn.tempname() .. '.vim' vim.cmd(('silent! mksession! %s'):format(session_file)) vim.g.cp_saved_session = session_file vim.cmd.diffoff() vim.cmd.only() vim.cmd.edit(temp_output) vim.cmd.diffthis() clearcol() vim.cmd.vsplit(expected) vim.cmd.diffthis() clearcol() vim.cmd(('botright split %s'):format(input)) clearcol() vim.cmd.wincmd('k') vim.g.cp_diff_mode = true vim.g.cp_temp_output = temp_output log('entered diff mode') end end local initialized = false function M.setup(user_config) if initialized and not user_config then return end config = config_module.setup(user_config) local plugin_path = get_plugin_path() config.template_dir = plugin_path .. '/templates' config.snippets.path = plugin_path .. '/templates/snippets' snippets.setup(config) if initialized then return end initialized = true setup_python_env() vim.api.nvim_create_user_command('CP', function(opts) local args = opts.fargs if #args == 0 then log( 'Usage: :CP ', vim.log.levels.ERROR ) return end local cmd = args[1] if vim.tbl_contains(competition_types, cmd) then if args[2] then setup_contest(cmd) if (cmd == 'atcoder' or cmd == 'codeforces') and args[3] then setup_problem(args[2], args[3]) else setup_problem(args[2]) end else setup_contest(cmd) end elseif cmd == 'run' then run_problem() elseif cmd == 'debug' then debug_problem() elseif cmd == 'diff' then diff_problem() else if vim.g.cp_contest then if ( vim.g.cp_contest == 'atcoder' or vim.g.cp_contest == 'codeforces' ) and args[2] then setup_problem(cmd, args[2]) else setup_problem(cmd) end else log( 'no contest mode set. run :CP first or use full command', vim.log.levels.ERROR ) end end end, { nargs = '*', complete = function(ArgLead, CmdLine, ...) local commands = vim.list_extend( vim.deepcopy(competition_types), { 'run', 'debug', 'diff' } ) return vim.tbl_filter(function(cmd) return cmd:find(ArgLead, 1, true) == 1 end, commands) end, }) end return M