%% %% %CopyrightBegin% %% %% Copyright Ericsson AB 2009-2016. All Rights Reserved. %% %% Licensed under the Apache License, Version 2.0 (the "License"); %% you may not use this file except in compliance with the License. %% You may obtain a copy of the License at %% %% http://www.apache.org/licenses/LICENSE-2.0 %% %% Unless required by applicable law or agreed to in writing, software %% distributed under the License is distributed on an "AS IS" BASIS, %% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. %% See the License for the specific language governing permissions and %% limitations under the License. %% %% %CopyrightEnd% -module(ex_canvas_paint). -behaviour(wx_object). %% Client API -export([start/1]). %% wx_object callbacks -export([init/1, terminate/2, code_change/3, handle_info/2, handle_call/3, handle_cast/2, handle_event/2, handle_sync_event/3]). -include_lib("wx/include/wx.hrl"). -record(state, { parent, config, canvas, pen, brush, old_pos, bitmap }). start(Config) -> wx_object:start_link(?MODULE, Config, []). %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% init(Config) -> wx:batch(fun() -> do_init(Config) end). do_init(Config) -> Parent = proplists:get_value(parent, Config), Panel = wxPanel:new(Parent, []), %% Setup sizers MainSizer = wxBoxSizer:new(?wxVERTICAL), Sizer = wxStaticBoxSizer:new(?wxVERTICAL, Panel, [{label, "wxDC"}]), %% Create the window to paint on and make it repaint the whole window on resize Canvas = wxPanel:new(Panel, [{style, ?wxFULL_REPAINT_ON_RESIZE}]), wxPanel:setToolTip(Canvas, "Left-click and hold to draw something - release to stop drawing.\n" "Middle-click to fill with pink\n" "Middle-dclick to fill with white.\n" "Right-click to clear."), %% Create a wxPen and a WxBrush and set its colors to draw with Brush = wxBrush:new(?wxWHITE), Pen = wxPen:new(?wxBLACK, [{width, 2}]), PrintButton = wxButton:new(Panel, ?wxID_ANY, [{label, "Print"}]), Bitmap = wxBitmap:new(30,30), %% Add to sizers wxSizer:add(Sizer, Canvas, [{flag, ?wxEXPAND}, {proportion, 1}]), wxSizer:add(MainSizer, PrintButton, []), wxSizer:add(MainSizer, Sizer, [{flag, ?wxEXPAND}, {proportion, 1}]), wxPanel:connect(PrintButton, command_button_clicked), wxPanel:connect(Canvas, paint, [callback]), wxPanel:connect(Canvas, size), wxPanel:connect(Canvas, left_down), wxPanel:connect(Canvas, left_dclick), wxPanel:connect(Canvas, left_up), wxPanel:connect(Canvas, right_down), wxPanel:connect(Canvas, middle_down), wxPanel:connect(Canvas, middle_dclick), wxPanel:setSizer(Panel, MainSizer), wxSizer:layout(MainSizer), {Panel, #state{parent=Panel, config=Config, canvas = Canvas, pen = Pen, brush = Brush, bitmap = Bitmap}}. %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% %% Sync event from callback events, paint event must be handled in callbacks %% otherwise nothing will be drawn on windows. handle_sync_event(#wx{event = #wxPaint{}}, _wxObj, #state{canvas=Canvas, bitmap=Bitmap}) -> DC = wxPaintDC:new(Canvas), redraw(DC, Bitmap), wxPaintDC:destroy(DC), ok. %% Print what's drawn handle_event(#wx{event = #wxCommand{type = command_button_clicked}}, State = #state{bitmap=Bitmap}) -> PD = wxPrintData:new(), PDD = wxPrintDialogData:new(PD), PSDD = wxPageSetupDialogData:new(PD), Fun = fun(This,_Page) -> MX = MY = 500, wxPrintout:fitThisSizeToPageMargins(This, {MX,MY}, PSDD), {_X,_Y,W,H} = wxPrintout:getLogicalPageMarginsRect(This, PSDD), wxPrintout:offsetLogicalOrigin(This,(W-MX) div 2, (H-MY) div 2), DC = wxPrintout:getDC(This), redraw(DC, Bitmap), true end, Printout1 = wxPrintout:new("Print", Fun, [{getPageInfo, fun getPageInfo/1}]), Printout2 = wxPrintout:new("Print", Fun, [{getPageInfo, fun getPageInfo/1}]), Preview = wxPrintPreview:new(Printout1, [{printoutForPrinting,Printout2},{data,PDD}]), case wxPrintPreview:isOk(Preview) of true -> Env = wx:get_env(), spawn_link(fun() -> wx:set_env(Env), PF = wxPreviewFrame:new(Preview, State#state.parent, []), wxPreviewFrame:centre(PF, [{dir, ?wxBOTH}]), wxPreviewFrame:initialize(PF), wxPreviewFrame:centre(PF), wxPreviewFrame:show(PF) end); false -> io:format("Could not create preview window.\n" "Perhaps your current printer is not set correctly?~n", []), wxPrintPreview:destroy(Preview) end, {noreply, State#state{}}; %% Draw a line handle_event(#wx{event = #wxMouse{type = motion, x = X, y = Y}}, State = #state{canvas = Canvas, pen = Pen, brush = Brush}) -> Fun = fun(DC) -> wxDC:setPen(DC, Pen), wxBrush:setColour(Brush, ?wxBLACK), wxDC:setBrush(DC, Brush), wxDC:drawLine(DC, {X,Y}, State#state.old_pos) end, draw(Canvas,State#state.bitmap, Fun), {noreply, State#state{old_pos = {X,Y}}}; %% Resize event handle_event(#wx{event = #wxSize{size = {W,H}}}, State = #state{bitmap=Prev}) -> case W > 0 andalso H > 0 of true -> wxBitmap:destroy(Prev), Bitmap = wxBitmap:new(W,H), draw(State#state.canvas, Bitmap, fun(DC) -> wxDC:clear(DC) end), {noreply, State#state{bitmap=Bitmap}}; false -> {noreply, State} end; handle_event(#wx{event = #wxMouse{type = left_dclick,x = X,y = Y}}, State = #state{}) -> wxPanel:connect(State#state.canvas, motion), {noreply, State#state{old_pos = {X,Y}}}; handle_event(#wx{event = #wxMouse{type = left_down,x = X,y = Y}}, State = #state{}) -> wxPanel:connect(State#state.canvas, motion), {noreply, State#state{old_pos = {X,Y}}}; %% Fill with pink color handle_event(#wx{event = #wxMouse{type = middle_down,x = X, y =Y}}, State = #state{}) -> case os:type() of {_, darwin} -> io:format("Fill doesn't work on Darwin ~n",[]); _ -> ok end, Fun = fun(DC) -> wxBrush:setColour(State#state.brush, {255,125,255,255}), wxDC:setBrush(DC, State#state.brush), wxDC:floodFill(DC, {X,Y}, ?wxBLACK, [{style, ?wxFLOOD_BORDER}]) end, draw(State#state.canvas, State#state.bitmap, Fun), {noreply, State}; %% Fill with white color handle_event(#wx{event = #wxMouse{type = middle_dclick,x = X, y =Y}}, State = #state{}) -> Fun = fun(DC) -> wxBrush:setColour(State#state.brush, ?wxWHITE), wxDC:setBrush(DC, State#state.brush), wxDC:floodFill(DC, {X,Y}, ?wxBLACK, [{style, ?wxFLOOD_BORDER}]) end, draw(State#state.canvas, State#state.bitmap,Fun), {noreply, State}; handle_event(#wx{event = #wxMouse{type = left_up}}, State = #state{}) -> wxPanel:disconnect(State#state.canvas, motion), {noreply, State}; %% Clear the DC handle_event(#wx{event = #wxMouse{type = right_down}}, State = #state{}) -> draw(State#state.canvas, State#state.bitmap, fun(DC) -> wxDC:clear(DC) end), {noreply, State}. %% Callbacks handled as normal gen_server callbacks handle_info(Msg, State) -> demo:format(State#state.config, "Got Info ~p\n", [Msg]), {noreply, State}. handle_call(shutdown, _From, State=#state{parent=Panel}) -> wxPanel:destroy(Panel), {stop, normal, ok, State}; handle_call(Msg, _From, State) -> demo:format(State#state.config, "Got Call ~p\n", [Msg]), {reply,{error, nyi}, State}. handle_cast(Msg, State) -> io:format("Got cast ~p~n",[Msg]), {noreply,State}. code_change(_, _, State) -> {stop, ignore, State}. terminate(_Reason, _) -> ok. %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% %% Local functions %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% draw(Canvas, Bitmap, Fun) -> MemoryDC = wxMemoryDC:new(Bitmap), CDC = wxClientDC:new(Canvas), Fun(MemoryDC), wxDC:blit(CDC, {0,0}, {wxBitmap:getWidth(Bitmap), wxBitmap:getHeight(Bitmap)}, MemoryDC, {0,0}), wxClientDC:destroy(CDC), wxMemoryDC:destroy(MemoryDC). redraw(DC, Bitmap) -> try MemoryDC = wxMemoryDC:new(Bitmap), wxDC:blit(DC, {0,0}, {wxBitmap:getWidth(Bitmap), wxBitmap:getHeight(Bitmap)}, MemoryDC, {0,0}), wxMemoryDC:destroy(MemoryDC) catch error:{{badarg,_},_} -> %% Bitmap have been deleted ok end. getPageInfo(_This) -> {1,1,1,1}.