function [objective, TerminalReorder, TreeChildren] = QuantumMCMC(X, num_STdecomp)
% num_STdecomp: Number of particles for Suzuki-Trotter decomposition

figure('position',[100,100,600,600]);

maxiter = 20000;
% X: NumDat x Features
n = size(X,1);
D = squareform( pdist( X ) ); % Pair-wise dist. matrix
D = exp((-1)*D/10);

% Superpermutation for binary tree
SupTree = ones(1,(n-1)*(n-1));
for ii=1:n-1
    SupTree( (ii-1)*(n-1)+1:ii*(n-1) ) = randperm(n-1);
end

% Superpermutation for terminal nodes
SupTerminal = ones(1, n^2);
for ii= 1:n
    SupTerminal( (ii-1)*n+1:ii*n ) = randperm(n);
end

% Subsequence for binary tree
for mm=1:num_STdecomp
    SubIndTree{mm} = zeros(1,n-1);
    indSet = 1:(n-1)^2;
    for ii=1:n-1
        temp = indSet(SupTree==ii);
        temptemp = randi(length(temp));
        SubIndTree{mm}(ii) = temp(temptemp);
    end
    sigma{mm} = SupTree(sort(SubIndTree{mm},'ascend'));
end

% Subsequence for binary tree
for mm=1:num_STdecomp
    SubIndTerminal{mm} = zeros(1,n);
    indSet = 1:n^2;
    for ii=1:n
        temp = indSet(SupTerminal==ii);
        temptemp = randi(length(temp));
        SubIndTerminal{mm}(ii) = temp(temptemp);
    end
    TerminalReorder{mm} = SupTerminal(sort(SubIndTerminal{mm},'ascend'));
end

%%% MCMC initialization
for mm=1:num_STdecomp
    [TreeChildren{mm}, CountMat{mm}] = P2BT(sigma{mm});
    %objective = zeros(1,maxiter*2+1);
    objective{mm} = DasguptaScore(D, TerminalReorder{mm}, TreeChildren{mm}, CountMat{mm});
end

%%% MCMC main loop
for iter =1:maxiter
    for mm=1:num_STdecomp
        
        % Quantum dependencies
        if mm==1
            left_id = num_STdecomp;
        else
            left_id = mm-1;
        end
        if mm==num_STdecomp
            right_id = 1;
        else
            right_id = mm+1;
        end
        
        % Update binary tree
        [SubIndTree{mm},sigma{mm},TreeChildren{mm},CountMat{mm},objective{mm}]...
            =update_binarytree(n,SubIndTree{mm},sigma{mm},SupTree,TreeChildren{mm},...
            TerminalReorder{mm},CountMat{mm},objective{mm},D,...
            SubIndTree{left_id}, SubIndTree{right_id}, num_STdecomp);
        
        % Update Terminal nodel reordering
        [SubIndTerminal{mm},TerminalReorder{mm},objective{mm}]...
            =update_terminalnodepermutation(n,SubIndTerminal{mm},SupTerminal,...
            TreeChildren{mm},TerminalReorder{mm},CountMat{mm},objective{mm},D,...
            SubIndTerminal{left_id}, SubIndTerminal{right_id}, num_STdecomp);
    end
    
    % Visualization
    visQMCMC(n, objective, TreeChildren, D, TerminalReorder, sigma, num_STdecomp);
    %plot(cell2mat(objective));
    drawnow
    
end

end


function [SubIndTree,sigma,TreeChildren,CountMat,objective]...
    =update_binarytree(n,SubIndTree,sigma,SupTree,TreeChildren,TerminalReorder,CountMat,objective,D,...
    neighbor_left, neighbor_right, num_STdecomp)

% sanity_check
num_STdecomp=1;

indSet = 1:(n-1)^2;
% Update for Tree
id_update = randi(n-1);
new_SubIndTree = SubIndTree;
temp = indSet(SupTree==id_update);
temptemp = randi(length(temp));
new_SubIndTree(id_update) = temp(temptemp);
new_sigma = SupTree(sort(new_SubIndTree,'ascend'));
[new_TreeChildren, new_CountMat] = P2BT(new_sigma);
new_objective = DasguptaScore(D, TerminalReorder, new_TreeChildren, new_CountMat);

% Quantum effect
    q_const = num_STdecomp * log( 1 + n/( exp(n/num_STdecomp)-1 ));
    new_objective = new_objective...
        - q_const * ( sum(ismember(new_SubIndTree,neighbor_left))...
        +sum(ismember(new_SubIndTree,neighbor_right)) );

if rand(1) < exp(objective(end)-new_objective)
    SubIndTree = new_SubIndTree;
    sigma = new_sigma;
    TreeChildren = new_TreeChildren;
    CountMat = new_CountMat;
    objective = [objective;new_objective];
else
    objective = [objective;objective(end)];
end
end

function [SubIndTerminal,TerminalReorder,objective]...
    =update_terminalnodepermutation(n,SubIndTerminal,SupTerminal,TreeChildren,TerminalReorder,...
    CountMat,objective,D,...
    neighbor_left, neighbor_right, num_STdecomp)

% sanity_check
num_STdecomp=1;

indSet = 1:n^2;
% Update for TerminalReorder
    id_update = randi(n);
    new_SubIndTerminal = SubIndTerminal;
    temp = indSet(SupTerminal==id_update);
    temptemp = randi(length(temp));
    new_SubIndTerminal(id_update) = temp(temptemp);
    new_TerminalReorder = SupTerminal(sort(new_SubIndTerminal,'ascend'));
    new_objective = DasguptaScore(D, new_TerminalReorder, TreeChildren, CountMat);
    
    % Quantum effect
    q_const = num_STdecomp * log( 1 + (n-1)/( exp((n-1)/num_STdecomp)-1 ));
    new_objective = new_objective...
        - q_const * ( sum(ismember(new_SubIndTerminal,neighbor_left))...
        +sum(ismember(new_SubIndTerminal,neighbor_right)) );
    
    if rand(1) < exp(objective(end)-new_objective)
        SubIndTerminal = new_SubIndTerminal;
        TerminalReorder = new_TerminalReorder;
        objective = [objective;new_objective];
    else
        objective = [objective;objective(end)];
    end
end