/ src /
/src/NetworkUI.hs
1 {-# OPTIONS -cpp -fglasgow-exts #-}
2 module NetworkUI
3 ( create
4 , getConfig, Config
5 ) where
6
7 import SafetyNet
8 import State
9 import StateUtil
10 import Network
11 import Document
12 import INRule
13 import Common
14 import CommonIO
15 import qualified PersistentDocument as PD
16 import qualified PDDefaults as PD
17 import InfoKind
18 import DisplayOptions
19 import Constants
20 import Text.XML.HaXml.XmlContent (XmlContent)
21 import Text.Parse as Parse
22 import Operations
23 import INReduction
24 import INChecksUI
25 import CommonUI
26
27 import Graphics.UI.WX hiding (Child, upKey, downKey, swap)
28 import Graphics.UI.WXCore
29 import Functional.UI
30
31 data Config = NFC
32 { nfcWinDimensions :: (Int, Int, Int, Int) -- x, y, width, height
33 , nfcFileName :: Maybe String
34 , nfcSelection :: Document.Selection
35 }
36 deriving (Read, Show)
37
38 getConfig :: State g n e -> IO Config
39 getConfig state =
40 do{ theFrame <- getNetworkFrame state
41 ; (x, y) <- safeGetPosition theFrame
42 ; winSize <- get theFrame clientSize
43 ; pDoc <- getDocument state
44 ; maybeFileName <- PD.getFileName pDoc
45 ; doc <- PD.getDocument pDoc
46 ; return (NFC
47 { nfcWinDimensions = (x, y, sizeW winSize, sizeH winSize)
48 , nfcFileName = maybeFileName
49 , nfcSelection = getSelection doc
50 })
51 }
52
53
54 create :: (InfoKind n g, InfoKind e g, XmlContent g, Parse g, Show g) =>
55 State g n e -> g -> n -> e -> GraphOps g n e -> IO ()
56 create state g n e ops =
57
58 {- Containment structure of widgets:
59
60 * theFrame :: Frame ()
61 ** mainPan :: Panel ()
62 *** sp :: SplitterWindow ()
63 **** sp1 :: SplitterWindow ()
64 ***** sp2 :: SplitterWindow ()
65 ****** palettePan :: Panel ()
66 ******* palPan :: Panel ()
67 ******* addAgent :: Button ()
68 ****** rulesTreePan :: Panel ()
69 ******* tree :: TreeCtrl ()
70 ******* buttonNewRule :: Button ()
71 ******* buttonNewIRule :: Button ()
72 ***** sp3 :: SplitterWindow ()
73 ****** sp4 :: SplitterWindow ()
74 ******* ruleLHSPan :: Panel ()
75 ****** canvas
76 ******* ruleRHSPan :: Panel ()
77 ****** lhs2rhsB :: Button ()
78 ****** canvas
79 ****** netPan :: Panel ()
80 ******* canvas
81 ******* strategies :: RadioBox ()
82 ******* steps :: RadioBox ()
83 ******* reduceB :: Button ()
84 ******* reduceStopB :: Button ()
85 **** textlog :: TextCtrl ()
86 -}
87
88 do{ theFrame <- frame [ text := "Interaction Nets editor"
89 , position := pt 200 20
90 , clientSize := sz 300 240 ]
91 ; setNetworkFrame theFrame state
92
93 -- Panels and SplitterWindows
94 ; mainPan <- panel theFrame []
95 ; sp <- splitterWindow mainPan []
96 ; sp1 <- splitterWindow sp []
97 ; sp2 <- splitterWindow sp1 []
98 ; sp3 <- splitterWindow sp1 []
99
100 -- create a panel and put the Visible Palette with buttons on it
101 ; palettePan <- panel sp2 []
102 ; palPan <- panel palettePan []
103 ; setPalettePanel palPan state
104
105 ; rulesTreePan <- panel sp2 []
106 ; sp4 <- splitterWindow sp3 []
107 #if !defined(__APPLE__)
108 ; ruleLHSPan <- panel sp4 []
109 #endif
110 ; ruleRHSPan <- panel sp4 []
111
112 ; netPan <- panel sp3 []
113 -- Create page setup dialog and save in state
114 ; pageSetupData <- pageSetupDialogDataCreate
115 ; initialPageSetupDialog <- pageSetupDialogCreate theFrame pageSetupData
116 ; objectDelete pageSetupData
117 ; setPageSetupDialog initialPageSetupDialog state
118
119 -- Drawing area for net
120 ; let (width, height) = getCanvasSize (Network.empty g n e)
121 ; ppi <- getScreenPPI
122 ; canvas <- scrolledWindow netPan
123 [ virtualSize := sz (logicalToScreenX ppi width)
124 (logicalToScreenY ppi height)
125 , scrollRate := sz 10 10
126 , bgcolor := wxcolor paneBackgroundColor
127 , fullRepaintOnResize := False
128 ]
129 ; State.setCanvas canvas state
130
131 -- Dummy persistent document to pass around
132 ; pDoc <- getDocument state
133
134 -- Attach handlers to drawing area
135 ; set canvas
136 [ on paint := \dc _ -> safetyNet theFrame $ paintHandler state dc Net
137 , on mouse := \p -> safetyNet theFrame $
138 do setActiveCanvas Net state
139 mouseEvent p canvas theFrame state
140 --; focusOn canvas
141 , on keyboard := \k -> safetyNet theFrame $
142 do setActiveCanvas Net state
143 keyboardEvent theFrame state k
144 --; focusOn canvas
145 ]
146
147 -- Drawing area for LHS
148 ; let (width, height) = (100, 100) -- getCanvasSize (Network.empty g n e)
149 ; ppi <- getScreenPPI
150 #if !defined(__APPLE__)
151 ; canvasLHS <- scrolledWindow ruleLHSPan
152 #else
153 ; canvasLHS <- scrolledWindow sp4
154 #endif
155 [ virtualSize := sz (logicalToScreenX ppi width)
156 (logicalToScreenY ppi height)
157 , scrollRate := sz 10 10
158 , bgcolor := wxcolor paneBackgroundColor
159 , fullRepaintOnResize := False
160 ]
161 ; State.setLHSCanvas canvasLHS state
162
163 -- Attach handlers to drawing area
164 ; set canvasLHS
165 [ on paint := \dc _ -> safetyNet theFrame
166 $ do rule <- getActiveRule state
167 paintHandler state dc $ LHS rule
168 , on mouse := \p -> safetyNet theFrame $
169 do setActiveCanvas (LHS "") state
170 mouseEvent p canvasLHS theFrame state
171 --; focusOn canvasLHS
172 , on keyboard := \k -> safetyNet theFrame $
173 do setActiveCanvas (LHS "") state
174 keyboardEvent theFrame state k
175 --; focusOn canvasLHS
176 ]
177
178 -- buttons to copy LHS to RHS
179 ; lhs2rhsB <- button ruleRHSPan [ text := "==>"
180 , tooltip := "Copy the LHS network to the RHS."
181 , on command := safetyNet theFrame
182 $ lhs2rhsItem True state
183 ]
184 -- only copies the interface
185 ; lhsInt2rhsB <- button ruleRHSPan [ text := "=->"
186 , tooltip := "Copy the LHS interface to the RHS."
187 , on command := safetyNet theFrame
188 $ lhs2rhsItem False state
189 ]
190
191 -- Drawing area for RHS
192 ; let (width, height) = (100, 100) -- getCanvasSize (Network.empty g n e)
193 ; ppi <- getScreenPPI
194 ; canvasRHS <- scrolledWindow ruleRHSPan
195 [ virtualSize := sz (logicalToScreenX ppi width)
196 (logicalToScreenY ppi height)
197 , scrollRate := sz 10 10
198 , bgcolor := wxcolor paneBackgroundColor
199 , fullRepaintOnResize := False
200 ]
201 ; State.setRHSCanvas canvasRHS state
202
203 -- Attach handlers to drawing area
204 ; set canvasRHS
205 [ on paint := \dc _ -> safetyNet theFrame
206 $ do rule <- getActiveRule state
207 paintHandler state dc $ RHS rule
208 , on mouse := \p -> safetyNet theFrame $
209 do setActiveCanvas (RHS "") state
210 mouseEvent p canvasRHS theFrame state
211 --; focusOn canvasLHS
212 , on keyboard := \k -> safetyNet theFrame $
213 do setActiveCanvas (RHS "") state
214 keyboardEvent theFrame state k
215 --; focusOn canvasLHS
216 ]
217
218 -- to debug purposes
219 ; textlog <- textCtrlRich sp [enabled := False, wrap := WrapNone]
220 -- use text control as logger
221 ; textCtrlMakeLogActiveTarget textlog
222
223 -- File menu
224 ; fileMenu <- menuPane [ text := "&File" ]
225 ; menuItem fileMenu
226 [ text := "New\tCtrl+N"
227 , on command := safetyNet theFrame $ newItem state g n e
228 ]
229 ; menuItem fileMenu
230 [ text := "Open...\tCtrl+O"
231 , on command := safetyNet theFrame $ openItem theFrame state
232 >> singleCheckOverIN iNCheck state
233 ]
234 ; saveItem <- menuItem fileMenu
235 [ text := "Save\tCtrl+S"
236 , on command := safetyNet theFrame $ checkValidINOnSave state $ PD.save pDoc
237 ]
238 ; menuItem fileMenu
239 [ text := "Save as..."
240 , on command := safetyNet theFrame $ checkValidINOnSave state $ PD.saveAs pDoc
241 ]
242
243 ; menuLine fileMenu
244
245 ; menuItem fileMenu
246 [ text := "Page setup..."
247 , on command := safetyNet theFrame $
248 do{ psd <- getPageSetupDialog state
249 ; dialogShowModal psd
250 ; return ()
251 }
252 ]
253
254 ; menuItem fileMenu
255 [ text := "Print..."
256 , on command := safetyNet theFrame $
257 let printFun _ printInfo _ dc _ =
258 do { dcSetUserScale dc
259 (fromIntegral (sizeW (printerPPI printInfo))
260 / fromIntegral (sizeW (screenPPI printInfo)))
261 (fromIntegral (sizeH (printerPPI printInfo))
262 / fromIntegral (sizeH (screenPPI printInfo)))
263 ; paintHandler state dc Net
264 }
265 pageFun _ _ _ = (1, 1)
266 in
267 do{ psd <- getPageSetupDialog state
268 ; printDialog psd (toolName ++ " print") pageFun printFun
269 }
270 ]
271
272 ; menuItem fileMenu
273 [ text := "Print preview"
274 , on command := safetyNet theFrame $
275 let printFun _ _ _ dc _ = paintHandler state dc Net
276 pageFun _ _ _ = (1, 1)
277 in
278 do{ psd <- getPageSetupDialog state
279 ; printPreview psd (toolName ++ " preview") pageFun printFun
280 }
281 ]
282
283 ; menuLine fileMenu
284
285 ; menuItem fileMenu
286 [ text := "E&xit"
287 , on command := close theFrame
288 ]
289
290 -- Edit menu
291 ; editMenu <- menuPane [ text := "&Edit" ]
292 ; undoItem <- menuItem editMenu
293 [ on command := safetyNet theFrame $ do PD.undo pDoc; repaintAll state ]
294 ; redoItem <- menuItem editMenu
295 [ on command := safetyNet theFrame $ do PD.redo pDoc; repaintAll state ]
296 {-
297 ; menuLine editMenu
298 ; menuItem editMenu
299 [ text := "Edit global info..."
300 , on command := safetyNet theFrame $ changeGlobalInfo theFrame state
301 ]
302 -}
303
304 -- View menu
305 ; viewMenu <- menuPane [ text := "&View" ]
306 ; (DP opts) <- getDisplayOptions state
307 ; menuItem viewMenu
308 [ text := "Node Label"
309 , checkable := True
310 , checked := NodeLabel `elem` opts
311 , on command := safetyNet theFrame $ do
312 { changeDisplayOptions (toggle NodeLabel) state
313 ; repaintAll state } ]
314 {-
315 ; menuItem viewMenu
316 [ text := "Node Info"
317 , checkable := True
318 , checked := NodeInfo `elem` opts
319 , on command := safetyNet theFrame $ do
320 { changeDisplayOptions (toggle NodeInfo) state
321 ; repaintAll state } ]
322 -}
323 ; menuItem viewMenu
324 [ text := "Edge Label"
325 , checkable := True
326 , checked := EdgeLabel `elem` opts
327 , on command := safetyNet theFrame $ do
328 { changeDisplayOptions (toggle EdgeLabel) state
329 ; repaintAll state } ]
330 {-
331 ; menuItem viewMenu
332 [ text := "Edge Info"
333 , checkable := True
334 , checked := EdgeInfo `elem` opts
335 , on command := safetyNet theFrame $ do
336 { changeDisplayOptions (toggle EdgeInfo) state
337 ; repaintAll state } ]
338 -}
339
340 -- Operations menu
341 ; opsMenu <- menuPane [ text := "&Operations" ]
342 ; mapM_ (\ (name,_)->
343 menuItem opsMenu
344 [ text := name
345 , on command := safetyNet theFrame $ do
346 { callPureGraphOp name ops state
347 ; repaintAll state }
348 ]
349 ) (pureOps ops)
350 ; when (not . null $ pureOps ops)
351 $ menuLine opsMenu
352
353 ; mapM_ (\ (name,_)->
354 menuItem opsMenu
355 [ text := name
356 , on command := safetyNet theFrame $ do
357 { callIOGraphOp name ops state
358 ; repaintAll state }
359 ]
360 ) (ioOps ops)
361 ; when (not . null $ ioOps ops)
362 $ menuLine opsMenu
363
364 ; menuItem opsMenu
365 [ text := "Functional term to IN system"
366 , on command := safetyNet theFrame $ compileUI state g n e
367 ]
368
369 -- Palette menu
370 ; palMenu <- menuPane [ text := "&Symbols" ]
371 ; menuItem palMenu
372 [ text := "Save palette as ..."
373 , on command := safetyNet theFrame $ savePalette theFrame state
374 ]
375 ; menuItem palMenu
376 [ text := "Change shape palette..."
377 , on command := safetyNet theFrame $
378 do yes <- confirmDialog theFrame "Dangerous operation!!" "Changing the palette can lead to severe errors.\nUnless you know what you are doing, don't proceed.\nDo you really want to proceed ?" False
379 when yes $ openPalette theFrame state
380 ]
381
382 -- Checks menu
383 ; chksMenu <- menuPane [text := "&Checks"]
384 ; mapM_ (\ chk@(name, desc,_,_,_) ->
385 menuItem chksMenu [ text := name
386 , on command := safeAndClear theFrame textlog $ singleCheckOverIN chk state
387 ]
388 ) checksList
389 ; menuLine chksMenu
390 ; menuItem chksMenu [ text := "Multiple checks at once"
391 , on command := safeAndClear theFrame textlog $ multipleChecksOverIN_UI theFrame state
392 ]
393 ; menuItem chksMenu [ text := "Checks over multiple files"
394 , on command := safeAndClear theFrame textlog $ multipleChecksOverINs_UI state
395 ]
396
397 -- Help menu
398 ; helpMenu <- menuPane [ text := "&Help" ]
399 ; menuItem helpMenu
400 [ text := "How to use?"
401 , on command := createHelpWindow
402 ]
403 ; menuItem helpMenu
404 [ text := "About"
405 , on command := createAboutWindow theFrame
406 ]
407
408 ; PD.initialise pDoc (PD.PD
409 { PD.document = Document.empty g n e
410 , PD.history = []
411 , PD.future = []
412 , PD.limit = Nothing
413 , PD.fileName = Nothing
414 , PD.dirty = False
415 , PD.saveToDisk = saveToDisk theFrame
416 , PD.updateUndo = PD.defaultUpdateUndo undoItem
417 , PD.updateRedo = PD.defaultUpdateRedo redoItem
418 , PD.updateSave = PD.defaultUpdateSave saveItem
419 , PD.updateTitleBar = PD.defaultUpdateTitlebar theFrame toolName
420 , PD.saveChangesDialog = PD.defaultSaveChangesDialog theFrame toolName
421 , PD.saveAsDialog = PD.defaultSaveAsDialog theFrame extensions
422 })
423
424 ; setInterfacePalette n state
425
426 ; initializeRules state g n e
427 -- Rules Panel
428 ; tree <- treeCtrl rulesTreePan [style :~ (wxTR_EDIT_LABELS .+.)]
429 ; setTree tree state
430 ; top <- treeCtrlAddRoot tree "Rules" noImage noImage objectNull
431 -- ; treeCtrlSetItemClientData tree top (return ()) ""
432 ; addRules2Tree tree top state
433 ; treeCtrlExpand tree top
434 ; set tree [ on treeEvent := onTreeEvent tree state g n e]
435
436 ; buttonNewRule <- button rulesTreePan
437 [ text := "Add new rule"
438 , on command := safetyNet theFrame $ addNewRuleItem True state $ initial g n e
439 ]
440 ; buttonNewIRule <- button rulesTreePan
441 [ text := "Rule creation wizard"
442 , on command := safetyNet theFrame $ createRuleItem theFrame state g n e
443 ]
444
445
446 ; addAgent <- button palettePan [ text := "Create new symbol"
447 , on command := createNewAgentItem state
448 ]
449
450 ; set palettePan [ layout := column 5 [ widget addAgent
451 , hfloatCentre $ widget palPan ] ]
452
453
454 -- reduction
455 ; reduceB <- button netPan
456 [ text := "Reduce"
457 , enabled := True
458 ]
459 ; setReduceButton reduceB state
460
461 ; reduceStopB <- button netPan
462 [ text := "Stop"
463 , enabled := False
464 ]
465 ; let (stratsS, _) = unzip strategiesList
466
467 ; strategies <- radioBox netPan Vertical stratsS
468 [ text := "Strategy"
469 , selection := 1 ]
470
471 ; steps <- radioBox netPan Vertical ["one", "many"]
472 [ text := "Steps"
473 , selection := 0 ]
474
475 ; set reduceB [on command := do
476 set reduceB [enabled := False]
477 set reduceStopB [enabled := True]
478 set strategies [enabled := False]
479 set steps [enabled := False]
480 setContinueReduction True state
481 s <- get steps selection
482 i <- get strategies selection
483 globalReduce (stratsS !! i) (s==1) state
484 set reduceStopB [enabled := False]
485 set strategies [enabled := True]
486 set steps [enabled := (stratsS !! i /= "Manual selection")]
487 set reduceB [enabled := True]
488 ]
489
490 ; set reduceStopB [ on command := do stopReduction state
491 set reduceStopB [enabled := False]
492 set reduceB [enabled := True]
493 ]
494
495 ; set strategies [ on select :=
496 do i <- get strategies selection
497 if stratsS !! i == "Manual selection"
498 then set steps [ enabled := False
499 , selection := 0
500 ]
501 else do set steps [ enabled := True]
502 set reduceB [ enabled := True]
503
504 ]
505
506
507 -- Layout the main window
508 ; set theFrame
509 [ menuBar := [ fileMenu, editMenu, viewMenu, opsMenu, palMenu, chksMenu, helpMenu ]
510 , layout := container mainPan $ margin 5 $ fill $
511 hsplit sp 5 500 ( -- to remove. just for debug
512 vsplit sp1 5 200
513 (hsplit sp2 5 200
514 (widget palettePan)
515 (container rulesTreePan $
516 column 5 [ fill $ widget tree
517 , hfloatCentre $ widget buttonNewRule
518 , hfloatCentre $ widget buttonNewIRule
519
520 ]
521 )
522 )
523 (hsplit sp3 5 250
524 (vsplit sp4 5 300
525 (
526 #if !defined(__APPLE__)
527 container ruleLHSPan $ boxed "LHS" $
528 #endif
529 fill $ widget canvasLHS)
530 (container ruleRHSPan $ row 5 [ vstretch $ valignCenter $ column 5 [ widget lhs2rhsB
531 , widget lhsInt2rhsB ]
532 ,
533 #if !defined(__APPLE__)
534 boxed "RHS" $
535 #endif
536 fill $ widget canvasRHS
537 ] ) )
538 (container netPan $
539 column 5 [ fill $ widget canvas
540 ,
541 #if !defined(__APPLE__)
542 boxed "Reduction" $
543 #endif
544 hfloatLeft $ row 5 [ widget strategies
545 , widget steps
546 , column 5 [ widget reduceB
547 , widget reduceStopB]
548 ]
549 ]
550 )
551 )
552 )(widget textlog) -- to remove. just for debug
553
554
555 , clientSize := sz 900 600
556 , statusBar := []
557 , on closing := safetyNet theFrame $ checkValidINOnSave state $ exit state
558 ]
559
560 -- ; set theFrame
561 -- [ position := pt 200 20
562 -- , clientSize := sz 300 240
563 -- ]
564 }
565