//#pragma GCC optimize("Ofast,unroll-loops")
//#pragma GCC target("avx2,tune=native")
#include <bits/stdc++.h>
using namespace std;
#define file "o"
#define ff(i, a, b) for(auto i=(a); i<=(b); ++i)
#define ffr(i, b, a) for(auto i=(b); i>=(a); --i)
#define nl "\n"
#define ss " "
#define pb emplace_back
#define fi first
#define se second
#define sz(s) (int)s.size()
#define all(s) (s).begin(), (s).end()
#define ms(a,x) memset(a, x, sizeof (a))
#define cn continue
#define re exit(0)
typedef long long ll;
typedef vector<int> vi;
const int maxn = 400000 + 15;
const ll INF = (ll)4e18;
int n, m;
int st[maxn], ed[maxn], timer=0;
int st2[maxn], tour2[20][maxn*2], timer2=0;
int h[maxn];
vi g[maxn], ng[maxn], c[maxn], cur;
ll ans[maxn], dff[maxn];
int node_at_tin[maxn];
// compute subtree size quickly as ed - st + 1
void dfs(int u, int par)
{
st[u]=++timer;
st2[u]=++timer2; tour2[0][timer2]=u;
for(auto v:g[u]) if(v!=par)
{
h[v]=h[u]+1;
dfs(v, u);
tour2[0][++timer2]=u;
}
ed[u]=timer;
}
void pre()
{
int LOG = 19;
ff(j, 1, LOG) ff(i, 1, timer2-(1<<j)+1)
{
int u=tour2[j-1][i], v=tour2[j-1][i+(1<<(j-1))];
if(h[u]<h[v]) tour2[j][i]=u;
else tour2[j][i]=v;
}
}
inline int lca(int u, int v)
{
int l=st2[u], r=st2[v];
if (l>r) swap(l, r);
int k=31-__builtin_clz(r-l+1);
int a = tour2[k][l], b = tour2[k][r-(1<<k)+1];
if(h[a] < h[b]) return a;
else return b;
}
inline bool cmp_tin(int u, int v)
{
return st[u] < st[v];
}
inline bool child_of(int u, int v)
{
return st[u] <= st[v] && ed[v] <= ed[u];
}
inline void upd(int l, int r, ll val)
{
if (l>r) return;
dff[l] += val;
dff[r+1] -= val;
}
void build_virtual()
{
sort(all(cur), cmp_tin);
int k = sz(cur);
// add LCAs of consecutive
for (int i = 0; i+1 < k; ++i) cur.pb(lca(cur[i], cur[i+1]));
sort(all(cur), cmp_tin);
cur.erase(unique(all(cur)), cur.end());
// clear adjacency
for (int x : cur) ng[x].clear();
// build with stack
vector<int> stck;
stck.reserve(sz(cur));
stck.push_back(cur[0]);
for (int i = 1; i < sz(cur); ++i)
{
int v = cur[i];
while (!child_of(stck.back(), v)) {
int u = stck.back(); stck.pop_back();
ng[stck.back()].pb(u);
}
stck.push_back(v);
}
while (sz(stck) > 1) {
int u = stck.back(); stck.pop_back();
ng[stck.back()].pb(u);
}
}
int cnt_in_subtree(const vi &vec, int v)
{
// vec sorted by tin
auto lo = lower_bound(vec.begin(), vec.end(), st[v], [&](int node, int value){ return st[node] < value; });
auto hi = upper_bound(vec.begin(), vec.end(), ed[v], [&](int value, int node){ return value < st[node]; });
return int(hi - lo);
}
int main()
{
ios::sync_with_stdio(false);
cin.tie(nullptr); cout.tie(nullptr);
if (fopen(file".inp","r")) {
freopen(file".inp","r",stdin);
freopen(file".out","w",stdout);
}
int sub; if(!(cin>>sub)) return 0;
cin >> n >> m;
ff(i,1,n){
int x; cin >> x; c[x].pb(i);
}
ff(i,1,n-1){
int u,v,w; cin >> u >> v >> w;
g[u].pb(v); g[v].pb(u);
}
// preprocess
timer = 0; timer2 = 0;
h[1] = 0;
dfs(1, 0);
pre();
// build inverse tin
ff(i,1,n) node_at_tin[st[i]] = i;
// mark array for colored nodes per color
static int is_col[maxn];
ms(is_col, 0);
ms(dff, 0);
// process each color
for (int col = 1; col <= n; ++col)
{
if (c[col].empty()) continue;
// vec: original colored nodes sorted by tin
vi vec = c[col];
sort(all(vec), cmp_tin);
// prepare cur = vec (will be appended LCAs inside build_virtual)
cur.clear();
for (int x : vec) cur.pb(x);
// mark colored nodes
for (int x : vec) is_col[x] = 1;
// build virtual tree (ng adjacency filled, cur updated)
build_virtual();
// iterate virtual nodes in cur and compute contributions
for (int v : cur)
{
// sum sizes of virtual children
ll sum_ch = 0;
for (int ch : ng[v]) {
int subch = ed[ch] - st[ch] + 1;
sum_ch += subch;
}
bool v_is_col = is_col[v];
int sub_v = ed[v] - st[v] + 1;
if (!v_is_col)
{
// one block: subtree[v] minus children's subtrees
ll block_sz = (ll)sub_v - sum_ch;
if (block_sz > 0)
{
ll val = (ll)n - block_sz;
// add val to subtree[v]
upd(st[v], ed[v], val);
// subtract val from each child subtree
for (int ch : ng[v]) upd(st[ch], ed[ch], -val);
}
}
else
{
// v is colored: each neighbor-side that contains 0 colored nodes is a block
// iterate original neighbors
for (int to : g[v])
{
if (child_of(to, v)) {
// to is ancestor of v -> parent side; shouldn't happen because child_of(to,v) means to is ancestor of v
// but in tree adjacency, either to is child of v or parent of v
}
if (child_of(v, to)) {
// 'to' is child of v in original rooted tree
int cntcol = cnt_in_subtree(vec, to);
int side_sz = ed[to] - st[to] + 1;
if (cntcol == 0) {
ll val = (ll)n - side_sz;
upd(st[to], ed[to], val);
}
} else {
// 'to' is parent side (outside subtree v)
int total_col = sz(vec);
int inside_v = cnt_in_subtree(vec, v);
int outside_col = total_col - inside_v;
int side_sz = n - sub_v;
if (outside_col == 0 && side_sz > 0) {
ll val = (ll)n - side_sz;
// outside subtree: two intervals [1, st[v]-1] and [ed[v]+1, n]
if (st[v] > 1) upd(1, st[v]-1, val);
if (ed[v] < n) upd(ed[v]+1, n, val);
}
}
}
// colored node itself: every path to any node contains its own color
upd(st[v], st[v], (ll)n);
}
}
// cleanup marks
for (int x : vec) is_col[x] = 0;
}
// accumulate diff to ans
ll curv = 0;
ff(t,1,n)
{
curv += dff[t];
ans[node_at_tin[t]] = curv;
}
// print answers
ff(i,1,n)
{
if (i>1) cout << ' ';
cout << ans[i];
}
cout << '\n';
return 0;
}
//#pragma GCC optimize("Ofast,unroll-loops")
//#pragma GCC target("avx2,tune=native")
#include <bits/stdc++.h>
using namespace std;

#define file "o"
#define ff(i, a, b) for(auto i=(a); i<=(b); ++i)
#define ffr(i, b, a) for(auto i=(b); i>=(a); --i)
#define nl "\n"
#define ss " "
#define pb emplace_back
#define fi first
#define se second
#define sz(s) (int)s.size()
#define all(s) (s).begin(), (s).end()
#define ms(a,x) memset(a, x, sizeof (a))
#define cn continue
#define re exit(0)

typedef long long ll;
typedef vector<int> vi;

const int maxn = 400000 + 15;
const ll INF = (ll)4e18;

int n, m;
int st[maxn], ed[maxn], timer=0;
int st2[maxn], tour2[20][maxn*2], timer2=0;
int h[maxn];
vi g[maxn], ng[maxn], c[maxn], cur;
ll ans[maxn], dff[maxn];
int node_at_tin[maxn];

// compute subtree size quickly as ed - st + 1

void dfs(int u, int par)
{
    st[u]=++timer;
    st2[u]=++timer2; tour2[0][timer2]=u;
    for(auto v:g[u]) if(v!=par)
    {
        h[v]=h[u]+1;
        dfs(v, u);
        tour2[0][++timer2]=u;
    }
    ed[u]=timer;
}

void pre()
{
    int LOG = 19;
    ff(j, 1, LOG) ff(i, 1, timer2-(1<<j)+1)
    {
        int u=tour2[j-1][i], v=tour2[j-1][i+(1<<(j-1))];
        if(h[u]<h[v]) tour2[j][i]=u;
        else tour2[j][i]=v;
    }
}

inline int lca(int u, int v)
{
    int l=st2[u], r=st2[v];
    if (l>r) swap(l, r);
    int k=31-__builtin_clz(r-l+1);
    int a = tour2[k][l], b = tour2[k][r-(1<<k)+1];
    if(h[a] < h[b]) return a;
    else return b;
}

inline bool cmp_tin(int u, int v)
{
    return st[u] < st[v];
}

inline bool child_of(int u, int v)
{
    return st[u] <= st[v] && ed[v] <= ed[u];
}

inline void upd(int l, int r, ll val)
{
    if (l>r) return;
    dff[l] += val;
    dff[r+1] -= val;
}

void build_virtual()
{
    sort(all(cur), cmp_tin);
    int k = sz(cur);
    // add LCAs of consecutive
    for (int i = 0; i+1 < k; ++i) cur.pb(lca(cur[i], cur[i+1]));
    sort(all(cur), cmp_tin);
    cur.erase(unique(all(cur)), cur.end());
    // clear adjacency
    for (int x : cur) ng[x].clear();
    // build with stack
    vector<int> stck;
    stck.reserve(sz(cur));
    stck.push_back(cur[0]);
    for (int i = 1; i < sz(cur); ++i)
    {
        int v = cur[i];
        while (!child_of(stck.back(), v)) {
            int u = stck.back(); stck.pop_back();
            ng[stck.back()].pb(u);
        }
        stck.push_back(v);
    }
    while (sz(stck) > 1) {
        int u = stck.back(); stck.pop_back();
        ng[stck.back()].pb(u);
    }
}

int cnt_in_subtree(const vi &vec, int v)
{
    // vec sorted by tin
    auto lo = lower_bound(vec.begin(), vec.end(), st[v], [&](int node, int value){ return st[node] < value; });
    auto hi = upper_bound(vec.begin(), vec.end(), ed[v], [&](int value, int node){ return value < st[node]; });
    return int(hi - lo);
}

int main()
{
    ios::sync_with_stdio(false);
    cin.tie(nullptr); cout.tie(nullptr);
    if (fopen(file".inp","r")) {
        freopen(file".inp","r",stdin);
        freopen(file".out","w",stdout);
    }
    int sub; if(!(cin>>sub)) return 0;
    cin >> n >> m;
    ff(i,1,n){
        int x; cin >> x; c[x].pb(i);
    }
    ff(i,1,n-1){
        int u,v,w; cin >> u >> v >> w;
        g[u].pb(v); g[v].pb(u);
    }

    // preprocess
    timer = 0; timer2 = 0;
    h[1] = 0;
    dfs(1, 0);
    pre();
    // build inverse tin
    ff(i,1,n) node_at_tin[st[i]] = i;

    // mark array for colored nodes per color
    static int is_col[maxn];
    ms(is_col, 0);
    ms(dff, 0);
    // process each color
    for (int col = 1; col <= n; ++col)
    {
        if (c[col].empty()) continue;
        // vec: original colored nodes sorted by tin
        vi vec = c[col];
        sort(all(vec), cmp_tin);
        // prepare cur = vec (will be appended LCAs inside build_virtual)
        cur.clear();
        for (int x : vec) cur.pb(x);
        // mark colored nodes
        for (int x : vec) is_col[x] = 1;
        // build virtual tree (ng adjacency filled, cur updated)
        build_virtual();
        // iterate virtual nodes in cur and compute contributions
        for (int v : cur)
        {
            // sum sizes of virtual children
            ll sum_ch = 0;
            for (int ch : ng[v]) {
                int subch = ed[ch] - st[ch] + 1;
                sum_ch += subch;
            }
            bool v_is_col = is_col[v];
            int sub_v = ed[v] - st[v] + 1;
            if (!v_is_col)
            {
                // one block: subtree[v] minus children's subtrees
                ll block_sz = (ll)sub_v - sum_ch;
                if (block_sz > 0)
                {
                    ll val = (ll)n - block_sz;
                    // add val to subtree[v]
                    upd(st[v], ed[v], val);
                    // subtract val from each child subtree
                    for (int ch : ng[v]) upd(st[ch], ed[ch], -val);
                }
            }
            else
            {
                // v is colored: each neighbor-side that contains 0 colored nodes is a block
                // iterate original neighbors
                for (int to : g[v])
                {
                    if (child_of(to, v)) {
                        // to is ancestor of v -> parent side; shouldn't happen because child_of(to,v) means to is ancestor of v
                        // but in tree adjacency, either to is child of v or parent of v
                    }
                    if (child_of(v, to)) {
                        // 'to' is child of v in original rooted tree
                        int cntcol = cnt_in_subtree(vec, to);
                        int side_sz = ed[to] - st[to] + 1;
                        if (cntcol == 0) {
                            ll val = (ll)n - side_sz;
                            upd(st[to], ed[to], val);
                        }
                    } else {
                        // 'to' is parent side (outside subtree v)
                        int total_col = sz(vec);
                        int inside_v = cnt_in_subtree(vec, v);
                        int outside_col = total_col - inside_v;
                        int side_sz = n - sub_v;
                        if (outside_col == 0 && side_sz > 0) {
                            ll val = (ll)n - side_sz;
                            // outside subtree: two intervals [1, st[v]-1] and [ed[v]+1, n]
                            if (st[v] > 1) upd(1, st[v]-1, val);
                            if (ed[v] < n) upd(ed[v]+1, n, val);
                        }
                    }
                }
                // colored node itself: every path to any node contains its own color
                upd(st[v], st[v], (ll)n);
            }
        }
        // cleanup marks
        for (int x : vec) is_col[x] = 0;
    }

    // accumulate diff to ans
    ll curv = 0;
    ff(t,1,n)
    {
        curv += dff[t];
        ans[node_at_tin[t]] = curv;
    }

    // print answers
    ff(i,1,n)
    {
        if (i>1) cout << ' ';
        cout << ans[i];
    }
    cout << '\n';
    return 0;
}
