function [cost, x, d, t] = plotprofile(problem, x, d, t)
% Plot the cost function along a geodesic or a retraction path.
%
% function plotprofile(problem)
% function plotprofile(problem, x)
% function plotprofile(problem, x, d)
% function plotprofile(problem, x, d, t)
% function plotprofile(problem, x, [], t)
% function plotprofile(problem, [], [], t)
%
% function [costs, x, d, t] = plotprofile(...)
%
% Plot profile evaluates the cost function along a geodesic gamma(t) such
% that gamma(0) = x and the derivative of gamma at 0 is the direction d.
% The input t is a vector specifying for which values of t we must evaluate
% f(gamma(t)) (it may include negative values).
%
% If the function is called with an output, the plot is not drawn and the
% values of the cost are returned for the instants t.
%
% If x is omitted, a random point is picked. If d is omitted, a random
% unit-norm tangent vector at x is picked. If t is omitted, it is generated
% as a linspace over [-1, 1].
%
% See also surfprofile

% This file is part of Manopt: www.manopt.org.
% Original author: Nicolas Boumal, Jan. 9, 2013.
% Contributors: 
% Change log: 
%
%   April 3, 2015 (NB):
%       Works with the new StoreDB class system.
%
%   Nov. 12, 2016 (NB):
%       Making more inputs optional.
%
%   Nov.  9, 2022 (NB):
%       When d is generated at random, we force it to have norm 1.

    % Verify that the problem description is sufficient.
    if ~canGetCost(problem)
        error('It seems no cost function was provided.');  
    end
    
    if ~exist('x', 'var') || isempty(x)
        x = problem.M.rand();
        if exist('d', 'var') && ~isempty(d)
            error('If x is omitted, d should not be specified.');
        end
    end
    if ~exist('d', 'var') || isempty(d)
        d = problem.M.randvec(x);
        d = problem.M.lincomb(x, 1/problem.M.norm(x, d), d);
    end
    if ~exist('t', 'var') || isempty(t)
        t = linspace(-1, 1, 101);
    end
    
    if isfield(problem.M, 'exp')
        expo = problem.M.exp;
        str = 'Exp';
    else
        expo = problem.M.retr;
        str = 'Retr';
    end
    
    storedb = StoreDB();
    linesearch_fun = @(t) getCost(problem, expo(x, d, t), storedb);
    
    cost = zeros(size(t));
    for i = 1 : numel(t)
        cost(i) = linesearch_fun(t(i));
    end
    
    if nargout == 0
        plot(t, cost);
        xlabel('t');
        ylabel(['f(' str '_x(t*d))']);
    end
    
end
