#include <stdio.h>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <climits>
#include <cfloat>
#include <map>
#include <utility>
#include <set>
#include <iostream>
#include <memory>
#include <string>
#include <vector>
#include <algorithm>
#include <functional>
#include <sstream>
#include <complex>
#include <stack>
#include <queue>
#include <numeric>
#include <stdio.h>
#include <string.h>
#include <assert.h>
using namespace std;
static const double EPS = 1e-9;
int ROUND(double x) { return (int)(x+0.5); }
bool ISINT(double x) { return fabs(ROUND(x)-x)<=EPS; }
bool ISEQUAL(double x,double y) { return fabs(x-y)<=EPS*max(1.0,max(fabs(x),fabs(y))); }
double SQSUM(double x,double y) { return x*x+y*y; }
template<class T> bool INRANGE(T x,T a,T b) { return a<=x&&x<=b; }
#define PI (acos(-1))
#define ARRAY_NUM(a) (sizeof(a)/sizeof(a[0]))
#define NG (-1)
#define BIG (123456789)
#define BIGBIG (987654321987654321LL)
#define SZ(a) ((int)((a).size()))
#define SQ(a) ((a)*(a))
typedef long long ll;
typedef unsigned long long ull;
#define FOR(v,i) for(__typeof((v).begin())i=(v).begin();i!=(v).end();++i)
// BEGIN CUT HERE
#undef FOR
#define FOR(v,i) for(auto i=(v).begin();i!=(v).end();++i)
// END CUT HERE
class Dinic
{
public:
Dinic(int input_maxv) : maxv(input_maxv)
{
G.resize(input_maxv);
level.resize(input_maxv);
iter.resize(input_maxv);
}
void add_edge_both(int from, int to, int cap)
{
const int rev_from = SZ(G[from]);
const int rev_to = SZ(G[to]);
G[from].push_back(edge(to,cap,rev_to));
G[to].push_back(edge(from,cap,rev_from));
}
void add_edge(int from, int to, int cap)
{
const int rev_from = SZ(G[from]);
const int rev_to = SZ(G[to]);
G[from].push_back(edge(to,cap,rev_to));
G[to].push_back(edge(from,0,rev_from));
}
// sããtã¸ã®æ大æµãæ±ãã
int max_flow(int s, int t)
{
int flow = 0;
for(;;)
{
bfs(s);
if(level[t]<0) break;
fill(iter.begin(),iter.end(),0);
int f;
while( (f=dfs(s,t,DINIC_INF))>0)
{
flow += f;
}
}
return flow;
}
// ãã¼ãsãã辿ããç¯å²ãæ±ããï¼ãã以ä¸æµããªãã¨ããcap=0ã¯ããªã³ã¯ããªããªãï¼
// ï¼æµãçµãã£ããã¨sourceãããã©ããç¯å²ããæå°ã«ããæã®så´ããã©ããªãæ³ãtå´ããã®å¢çãã«ããããã¨ãããï¼
vector <bool> get_nodes_in_group(int s)
{
vector <bool> ret(maxv);
queue<int> que;
que.push(s);
while(!que.empty())
{
int v = que.front();
que.pop();
ret[v]=true;
for(int i=0;i<SZ(G[v]);i++)
{
edge &e = G[v][i];
if(e.cap>0 && !ret[e.to])
{
que.push(e.to);
}
}
}
return ret;
}
void disp()
{
for (int v = 0; v < maxv; v++)
{
printf("%d:",v);
for(int i=0;i<SZ(G[v]);i++)
{
if(G[v][i].init_cap>0)
{
printf("->%d(%d),",G[v][i].to,G[v][i].init_cap);
}
}
printf("\n");
}
}
private:
// sããã®æçè·é¢ãBFSã§è¨ç®ãã
void bfs(int s)
{
fill(level.begin(),level.end(),NG);
queue<int> que;
level[s]=0;
que.push(s);
while(!que.empty())
{
int v = que.front();
que.pop();
for(int i=0;i<SZ(G[v]);i++)
{
edge &e = G[v][i];
if(e.cap>0 && level[e.to]<0)
{
level[e.to] = level[v] + 1;
que.push(e.to);
}
}
}
}
// å¢å ãã¹ãDFSã§æ¢ã
int dfs(int v, int t, int f)
{
if(v==t) return f;
for (int &i=iter[v];i<SZ(G[v]);i++)
{
edge& e = G[v][i];
if(e.cap>0 && level[v]<level[e.to])
{
int d = dfs(e.to, t, min(f, e.cap));
if(d>0)
{
e.cap -= d;
G[e.to][e.rev].cap += d;
return d;
}
}
}
return 0;
}
static const int DINIC_INF = INT_MAX; // 容éãllã«ãããã¨ãã¯ããããå¤ãã
struct edge
{
edge(int input_to, int input_cap, int input_rev) : to(input_to), cap(input_cap), rev(input_rev), init_cap(input_cap) {}
int to; // è¡å
int cap; // 容é
int rev; // é辺
int init_cap; // åæ容éï¼ãããã°ç¨ï¼
};
int maxv;
vector < vector <edge> > G; // ã°ã©ãã®é£æ¥ãªã¹ã
vector < int > level; // sããã®è·é¢
vector < int > iter; // ã©ãã¾ã§èª¿ã¹çµãã£ãã
};
int main()
{
int T;
cin >> T;
for (int testcase = 0; testcase < T; testcase++)
{
int M,N;
cin >> M >> N;
const int S = M*N;
const int T = S+1;
const int V = T+1;
vector <string> vs;
for (int y = 0; y < M; y++)
{
string s;
cin >> s;
vs.push_back(s);
}
const static int dy[] = {-1, 0, 1, 0}; // U,R,D,L
const static int dx[] = { 0, 1, 0,-1};
Dinic* dinic = new Dinic(V);
int default_score = 0;
for (int y = 0; y < M; y++)
{
for (int x = 0; x < N; x++)
{
const int now = y*N+x;
if((x+y)%2==0)
{
if(vs[y][x]=='?')
{
default_score += 4;
// dinic->add_edge(S,now,0);
dinic->add_edge(now,T,4);
for(int d = 0; d < 4; d++)
{
const int ny = y+dy[d];
const int nx = x+dx[d];
const int next = ny*N+nx;
if(INRANGE(ny,0,M-1)&&INRANGE(nx,0,N-1))
{
if(vs[ny][nx]=='#' || vs[ny][nx]=='?')
{
dinic->add_edge(next,now,2);
}
}
}
}
else if (vs[y][x]=='#')
{
default_score += 4;
dinic->add_edge(now,T,BIG);
for(int d = 0; d < 4; d++)
{
const int ny = y+dy[d];
const int nx = x+dx[d];
if(INRANGE(ny,0,M-1)&&INRANGE(nx,0,N-1))
{
if(vs[ny][nx]=='#')
{
default_score -= 2;
}
}
}
}
}
else
{
if(vs[y][x]=='?')
{
default_score += 4;
dinic->add_edge(S,now,4);
// dinic->add_edge(now,T,0);
for(int d = 0; d < 4; d++)
{
const int ny = y+dy[d];
const int nx = x+dx[d];
const int next = ny*N+nx;
if(INRANGE(ny,0,M-1)&&INRANGE(nx,0,N-1))
{
if(vs[ny][nx]=='#')
{
dinic->add_edge(now,next,2);
}
}
}
}
else if (vs[y][x]=='#')
{
default_score += 4;
dinic->add_edge(S,now,BIG);
}
}
}
}
printf("Case #%d: %d\n",testcase+1,default_score - dinic->max_flow(S,T));
delete dinic;
}
return 0;
}
#include <stdio.h>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <climits>
#include <cfloat>
#include <map>
#include <utility>
#include <set>
#include <iostream>
#include <memory>
#include <string>
#include <vector>
#include <algorithm>
#include <functional>
#include <sstream>
#include <complex>
#include <stack>
#include <queue>
#include <numeric>
#include <stdio.h>
#include <string.h>
#include <assert.h>

using namespace std;
static const double EPS = 1e-9;
int ROUND(double x) { return (int)(x+0.5); }
bool ISINT(double x) { return fabs(ROUND(x)-x)<=EPS; }
bool ISEQUAL(double x,double y) { return fabs(x-y)<=EPS*max(1.0,max(fabs(x),fabs(y))); }
double SQSUM(double x,double y) { return x*x+y*y; }
template<class T> bool INRANGE(T x,T a,T b) { return a<=x&&x<=b; }
#define PI	(acos(-1))
#define ARRAY_NUM(a) (sizeof(a)/sizeof(a[0])) 
#define NG (-1)
#define BIG (123456789)
#define BIGBIG (987654321987654321LL)
#define SZ(a) ((int)((a).size()))
#define SQ(a) ((a)*(a))
typedef long long ll;
typedef unsigned long long ull;

#define FOR(v,i) for(__typeof((v).begin())i=(v).begin();i!=(v).end();++i)
// BEGIN CUT HERE
#undef FOR
#define FOR(v,i) for(auto i=(v).begin();i!=(v).end();++i)
// END CUT HERE

 
class Dinic
{
public:
    Dinic(int input_maxv) : maxv(input_maxv)
    {
        G.resize(input_maxv);
        level.resize(input_maxv);
        iter.resize(input_maxv);
    }
 
    void add_edge_both(int from, int to, int cap)
    {
        const int rev_from  = SZ(G[from]);
        const int rev_to    = SZ(G[to]);
        G[from].push_back(edge(to,cap,rev_to));
        G[to].push_back(edge(from,cap,rev_from));
    }
 
    void add_edge(int from, int to, int cap)
    {
        const int rev_from  = SZ(G[from]);
        const int rev_to    = SZ(G[to]);
        G[from].push_back(edge(to,cap,rev_to));
        G[to].push_back(edge(from,0,rev_from));
    }
 
    // sããtã¸ã®æå¤§æµãæ±ãã
    int max_flow(int s, int t)
    {
        int flow = 0;
        for(;;)
        {
            bfs(s);
            if(level[t]<0) break;
            fill(iter.begin(),iter.end(),0);
            int f;
            while( (f=dfs(s,t,DINIC_INF))>0)
            {
                flow += f;
            }
        }
 
        return flow;
    }
 
    //  ãã¼ãsããè¾¿ããç¯å²ãæ±ããï¼ããä»¥ä¸æµããªãã¨ããcap=0ã¯ããªã³ã¯ããªããªãï¼
    // ï¼æµãçµãã£ããã¨sourceãããã©ããç¯å²ããæå°ã«ããæã®så´ããã©ããªãæ³ãtå´ããã®å¢çãã«ããããã¨ãããï¼
    vector <bool> get_nodes_in_group(int s)
    {
        vector <bool> ret(maxv);
 
        queue<int> que;
        que.push(s);
        while(!que.empty())
        {
            int v = que.front();
            que.pop();
            ret[v]=true;
 
            for(int i=0;i<SZ(G[v]);i++)
            {
                edge &e = G[v][i];
                if(e.cap>0 && !ret[e.to])
                {
                    que.push(e.to);
                }
            }
        }
        return ret;
    }
 
    void disp()
    {
        for (int v = 0; v < maxv; v++)
        {
            printf("%d:",v);
            for(int i=0;i<SZ(G[v]);i++)
            {
                if(G[v][i].init_cap>0)
                {
                    printf("->%d(%d),",G[v][i].to,G[v][i].init_cap);
                }
            }
            printf("\n");
        }
    }
 
private:
    // sããã®æç­è·é¢ãBFSã§è¨ç®ãã
    void bfs(int s)
    {
        fill(level.begin(),level.end(),NG);
        queue<int> que;
        level[s]=0;
        que.push(s);
        while(!que.empty())
        {
            int v = que.front();
            que.pop();
            for(int i=0;i<SZ(G[v]);i++)
            {
                edge &e = G[v][i];
                if(e.cap>0 && level[e.to]<0)
                {
                    level[e.to] = level[v] + 1;
                    que.push(e.to);
                }
            }
        }
    }
 
    // å¢å ãã¹ãDFSã§æ¢ã
    int dfs(int v, int t, int f)
    {
        if(v==t) return f;
        for (int &i=iter[v];i<SZ(G[v]);i++)
        {
            edge& e = G[v][i];
            if(e.cap>0 && level[v]<level[e.to])
            {
                int d = dfs(e.to, t, min(f, e.cap));
                if(d>0)
                {
                    e.cap -= d;
                    G[e.to][e.rev].cap += d;
                    return d;
                }
            }
        }
        return 0;
    }
 
    static const int DINIC_INF = INT_MAX; // å®¹éãllã«ãããã¨ãã¯ããããå¤ãã
 
    struct edge 
    {
        edge(int input_to, int input_cap, int input_rev) : to(input_to), cap(input_cap), rev(input_rev), init_cap(input_cap) {}
        int to;     // è¡å
        int cap;    // å®¹é
        int rev;    // éè¾º
        int init_cap; // åæå®¹éï¼ãããã°ç¨ï¼
    };
 
    int maxv;
    vector < vector <edge> > G; // ã°ã©ãã®é£æ¥ãªã¹ã
    vector < int > level;     // sããã®è·é¢
    vector < int > iter;      // ã©ãã¾ã§èª¿ã¹çµãã£ãã
 
};
 
int main()
{
	int T;
	cin >> T;

	for (int testcase = 0; testcase < T; testcase++)
	{
		int M,N;
		cin >> M >> N;

		const int S = M*N;
		const int T = S+1;
		const int V = T+1;
		vector <string> vs;
		for (int y = 0; y < M; y++)
		{
			string s;
			cin >> s;
			vs.push_back(s);
		}


		const static int dy[] = {-1, 0, 1, 0}; // U,R,D,L
		const static int dx[] = { 0, 1, 0,-1};

		Dinic* dinic = new Dinic(V);

		int default_score = 0;


		for (int y = 0; y < M; y++)
		{
			for (int x = 0; x < N; x++)
			{
				const int now = y*N+x;
				if((x+y)%2==0)
				{
					if(vs[y][x]=='?')
					{
						default_score += 4;
						//						dinic->add_edge(S,now,0);
						dinic->add_edge(now,T,4);

						for(int d = 0; d < 4; d++)
						{
							const int ny = y+dy[d]; 
							const int nx = x+dx[d];
							const int next = ny*N+nx;
							if(INRANGE(ny,0,M-1)&&INRANGE(nx,0,N-1))
							{
								if(vs[ny][nx]=='#' || vs[ny][nx]=='?')
								{
									dinic->add_edge(next,now,2);
								}
							}
						}
					}
					else if (vs[y][x]=='#')
					{
						default_score += 4;
						dinic->add_edge(now,T,BIG);

						for(int d = 0; d < 4; d++)
						{
							const int ny = y+dy[d]; 
							const int nx = x+dx[d];
							if(INRANGE(ny,0,M-1)&&INRANGE(nx,0,N-1))
							{
								if(vs[ny][nx]=='#')
								{
									default_score -= 2;
								}
							}
						}
					}
				}
				else
				{
					if(vs[y][x]=='?')
					{
						default_score += 4;

						dinic->add_edge(S,now,4);
						//						dinic->add_edge(now,T,0);

						for(int d = 0; d < 4; d++)
						{
							const int ny = y+dy[d]; 
							const int nx = x+dx[d];
							const int next = ny*N+nx;
							if(INRANGE(ny,0,M-1)&&INRANGE(nx,0,N-1))
							{
								if(vs[ny][nx]=='#')
								{
									dinic->add_edge(now,next,2);
								}
							}
						}
					}
					else if (vs[y][x]=='#')
					{
						default_score += 4;
						dinic->add_edge(S,now,BIG);
					}
				}
			}
		}


		printf("Case #%d: %d\n",testcase+1,default_score - dinic->max_flow(S,T));

		delete dinic;
	}


	return 0;
}
