12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396 |
- // GridCtrl.cpp : implementation file
- //
- // MFC Grid Control
- //
- // Written by Chris Maunder
- // mailto:chrismaunder@codeguru.com
- //
- // Copyright (c) 1998-1999.
- //
- // The code contained in this file is based on the original
- // WorldCom Grid control written by Joe Willcoxson,
- // mailto:chinajoe@aol.com
- // http://users.aol.com/chinajoe
- // The code has gone through so many modifications that I'm
- // not sure if there is even a single original line of code.
- // In any case Joe's code was a great framewaork on which to
- // build.
- //
- // This code may be used in compiled form in any way you desire. This
- // file may be redistributed unmodified by any means PROVIDING it is
- // not sold for profit without the authors written consent, and
- // providing that this notice and the authors name and all copyright
- // notices remains intact. If the source code in this file is used in
- // any commercial application then a statement along the lines of
- // "Portions copyright (c) Chris Maunder, 1998" must be included in
- // the startup banner, "About" box or printed documentation. An email
- // letting me know that you are using it would be nice as well. That's
- // not much to ask considering the amount of work that went into this.
- //
- // This file is provided "as is" with no expressed or implied warranty.
- // The author accepts no liability for any damage/loss of business that
- // this product may cause.
- //
- // Expect bugs!
- //
- // Please use and enjoy, and let me know of any bugs/mods/improvements
- // that you have found/implemented and I will fix/incorporate them into
- // this file.
- //
- // History:
- // --------
- // This control is constantly evolving, sometimes due to new features that I
- // feel are necessary, and sometimes due to existing bugs. Where possible I
- // have credited the changes to those who contributed code corrections or
- // enhancements (names in brackets) or code suggestions (suggested by...)
- //
- // 1.0 20 Feb 1998 First release version.
- // 1.01 24 Feb 1998 Memory leak fix (Jens Bohlmann)
- // Fixec typo (my fault!) in CMemDC.h - Claus Arend-Schneider)
- // Bug in GetSelectedCount (Lyn Newton)
- // 1.02 4 Mar 1998 Scrolling a little neater (less dead area)
- // Cell selection via OnTimer correctly updates Focus cell (Suggested by Lyn Newton)
- // 1.03 17 Mar 1998 Clipboard functions added, Intellimouse support
- // Using 32 bit scroll pos functions instead of 16 bit ("cronos")
- // Added OLE drag and drop.
- // 1.04 6 Apr 1998 Added Ctrl-A = Select All, fixed CGridDropTarget
- // problem, minor bug in CopyTextFromGrid (assert on
- // empty string). Cleaned up reponse to m_bEditable
- // (OnDrop and Ctrl-X disabled)
- // 1.05 10 May 1998 Memory leak fixed. (Yuheng Zhao)
- // Changed OLE initialisation (Carlo Comino)
- // Added separate fore + background cell colours (Suggested by John Crane)
- // ExpandToFit etc cleaned up - now decreases and
- // increases cell sizes to fit client area.
- // Added notification messages for the grid's parent (Suggested by
- // Added GVIS_READONLY state
- // 1.06 20 May 1998 Added TAB key handling. (Daniela Rybarova)
- // Intellimouse code correction for whole page scrolling (Paul Grant)
- // Fixed 16 bit thumb track problems (now 32 bit) (Paul Grant)
- // Fixed accelerator key problem in CInPlaceEdit (Matt Weagle)
- // Fixed Stupid ClassWizard code parsing problem (Michael A. Barnhart)
- // Double buffering now programmatically selectable
- // Workaround for win95 drag and drop registration problem
- // Corrected UNICODE implementation of clipboard stuff
- // Dragging and dropping from a selection onto itself no
- // no longer causes the cells to be emptied
- // 1.07 28 Jul 1998 Added EnsureVisible. (Roelf Werkman)
- // Fixed delete key problem on read-only cells. (Serge Weinstock)
- // OnEndInPlaceEdit sends notification AFTER storing
- // the modified text in the cell.
- // Added CreateInPlaceEditControl to make it easier to
- // change the way cells are edited. (suggested by Chris Clark)
- // Added Set/GetGridColor.
- // CopyTextToClipboard and PasteTextToGrid problem with
- // blank cells fixed, and CopyTextToClipboard tweaked.
- // SetModified called when cutting text or hitting DEL. (Jonathan Watters)
- // Focus cell made visible when editing begins.
- // Blank lines now treated correctly when pasting data.
- // Removed ES_MULTILINE style from the default edit control.
- // Added virtual CreateCell(row, col) function.
- // Fonts now specified on a per-cell basis using Get/SetItemFont.
- // 1.08 6 Aug 1998 Ctrl+arrows now allows cell navigation. Modified
- // CreateInPlaceEditControl to accept ID of control.
- // Added Titletips to grid cells. (Added EnableTitleTips / GetTitleTips)
- // 1.09 12 Sep 1998 When printing, parent window title is printed in header - Gert Rijs
- // GetNextItem search with GVNI_DROPHILITED now returns
- // cells with GVIS_DROPHILITED set, instead of GVIS_FOCUSED (Franco Bez)
- // (Also fixed minor bug in GetNextItem) (Franco Bez)
- // Cell selection using Shift+arrows works - Franco Bez
- // SetModified called after edits ONLY if contents changed (Franco Bez)
- // Cell colours now dithered in 256 colour screens.
- // Support for MSVC 4.2 (Graham Cheetham)
- // 1.10 30 Nov 1998 Titletips now disappear on a scroll event. Compiler errors
- // fixed. Grid lines drawing fixed (Graham Cheetham).
- // Cell focus fix on Isert Row/Col (Jochen Kauffmann)
- // Added DeleteNonFixedRows() (John Rackley)
- // Message #define conflict fixed (Oskar Wieland)
- // Titletips & cell insert/delete fix (Ramesh Dhar)
- // Titletips repeat-creation bug fixed.
- // GVN_SELCHANGED message now sends current cell ID
- // Font resource leak in GetTextExtent fixed (Gavin Jerman)
- // More TAB fixes (Andreas Ruh)
- // 1.11 1 Dec 1998 GetNextItem bug fix (suggested by Francis Fu)
- // InsertColumn (-1) fix (Roy Hopkins)
- // Was too liberal with the "IsEditable"'s. oops. (Michel Hete)
- // 1.11a 4 Jan 1999 Compiler errors in VC6 fixed.
- // 1.12 10 Apr 1999 Cleanup to allow GRIDCONTROL_NO_CLIPBOARD define
- // CE #defines added. (Thanks to Frank Uzzolino for a start on this)
- // TitleTip display fixed for cells with images, plus it now uses cell font
- // Added GetTextRect and IsCellFixed
- // Focus change problem when resizing columns fixed (Sergey Nikiforenko)
- // Grid line drawing problem in fixed cells fixed (Sergey Nikiforenko)
- // CreateCell format persistance bug fixed (Sergey Nikiforenko)
- // DeleteColumn now returns TRUE (oops) (R. Elmer)
- // Enter, Tab and Esc key problem (finally) fixed - Darren Webb and Koay Kah Hoe
- // OnSize infinite loop fixed - Steve Kowald
- // GVN_SELCHANGING and GVN_SELCHANGED values changed to avoid conflicts (Hiroaki Watanabe)
- // Added single row selection mode (Yao Cai)
- // Fixed image drawing clip problem
- // Reduced unnecessary redraws significantly
- // GetNextItem additions and bug fix, and GVNI_AREA search option (Franco Bez)
- // Added GVIS_MODIFIED style for cells, so individual cells can have their
- // modification status queried. (Franco Bez)
- // 1.12a 15 Apr 1999 Removed the SetModified/GetModified inlines (no compiler warning!)
- // Renamed IDC_INPLACE_CONTROL to IDC_INPLACE_CONTROL and moved
- // to the header
- //
- // TODO:
- // - OnOutOfMemory function instead of exceptions
- // - Decrease timer interval over time to speed up selection over time
- //
- // NOTE: Grid data is stored row-by-row, so all operations on large numbers
- // of cells should be done row-by-row as well.
- //
- // KNOWN ISSUES TO BE ADDRESSED (Please don't send bug reports):
- // * Killfocus comes to late when a command is selected by the Menu.
- // When you are editing a cell and choose a Menuitem that searches for all the
- // modified cells it is not found. When you chose the menu a second time it is
- // found. I assume that the Menu command is executed before the cell receives the
- // KillFocus event. Expect similar Problems with accelerators. (Franco Bez)
- // * When you select a cell and move the mouse around (with the Left button down
- // i.e continuing with your selection) - if the mouse is over the Fixed column
- // or Row the drawing of the selected region is strange - in particular as you
- // move up and down say the Left Fixed Column notice the behaviour of the Focus
- // Cell - it is out of sync. (Vinay Desai)
- //
- /////////////////////////////////////////////////////////////////////////////
-
- #include "stdafx.h"
- #include "MemDC.h"
- #include "GridCtrl.h"
- #include "InPlaceEdit.h"
- #include "ModeDlg.h"
- #include "ReliefDlg.h"
- // OLE stuff for clipboard operations
- #include <afxadv.h> // For CSharedFile
- #include <afxconv.h> // For LPTSTR -> LPSTR macros
- #ifdef _DEBUG
- #define new DEBUG_NEW
- #undef THIS_FILE
- static char THIS_FILE[] = __FILE__;
- #endif
- #define HEADER_HEIGHT 2 // For printing
- #define FOOTER_HEIGHT 2
- #define LEFT_MARGIN 4
- #define RIGHT_MARGIN 4
- #define TOP_MARGIN 1
- #define BOTTOM_MARGIN 1
- #define GAP 1
- #define SELECTED_CELL_FONT_WEIGHT 600 // weight of text for selected items
- IMPLEMENT_DYNCREATE(CGridCtrl, CWnd)
- void AFXAPI DDX_GridControl(CDataExchange* pDX, int nIDC, CGridCtrl& rControl)
- {
- if (rControl.GetSafeHwnd() == NULL) // not subclassed yet
- {
- ASSERT(!pDX->m_bSaveAndValidate);
- HWND hWndCtrl = pDX->PrepareCtrl(nIDC);
- if (!rControl.SubclassWindow(hWndCtrl))
- {
- ASSERT(FALSE); // possibly trying to subclass twice?
- AfxThrowNotSupportedException();
- }
- #ifndef _AFX_NO_OCC_SUPPORT
- else
- {
- // If the control has reparented itself (e.g., invisible control),
- // make sure that the CWnd gets properly wired to its control site.
- if (pDX->m_pDlgWnd->GetSafeHwnd() != ::GetParent(rControl.GetSafeHwnd()))
- rControl.AttachControlSite(pDX->m_pDlgWnd);
- }
- #endif //!_AFX_NO_OCC_SUPPORT
- }
- }
- // Get the number of lines to scroll with each mouse wheel notch
- // Why doesn't windows give us this function???
- UINT GetMouseScrollLines()
- {
- int nScrollLines = 3; // reasonable default
- HKEY hKey;
- if (RegOpenKeyEx(HKEY_CURRENT_USER, _T("Control Panel\\Desktop"),
- 0, KEY_QUERY_VALUE, &hKey) == ERROR_SUCCESS)
- {
- TCHAR szData[128];
- DWORD dwKeyDataType;
- DWORD dwDataBufSize = sizeof(szData);
- if (RegQueryValueEx(hKey, _T("WheelScrollLines"), NULL, &dwKeyDataType,
- (LPBYTE) &szData, &dwDataBufSize) == ERROR_SUCCESS)
- {
- nScrollLines = _tcstoul(szData, NULL, 10);
- }
- RegCloseKey(hKey);
- }
- return nScrollLines;
- }
- /////////////////////////////////////////////////////////////////////////////
- // CGridCtrl
- CGridCtrl::CGridCtrl(int nRows, int nCols, int nFixedRows, int nFixedCols)
- {
- RegisterWindowClass();
- nParentType = 0;
- bEditing = false;
- // Initialize OLE libraries
- m_bMustUninitOLE = FALSE;
- #if !defined(GRIDCONTROL_NO_DRAGDROP) || !defined(GRIDCONTROL_NO_CLIPBOARD)
- _AFX_THREAD_STATE* pState = AfxGetThreadState();
- if (!pState->m_bNeedTerm)
- {
- SCODE sc = ::OleInitialize(NULL);
- if (FAILED(sc))
- AfxMessageBox(_T("OLE initialization failed. Make sure that the OLE libraries are the correct version"));
- else
- m_bMustUninitOLE = TRUE;
- }
- #endif
- // Store the system colours in case they change. The gridctrl uses
- // these colours, and in OnSysColorChange we can check to see if
- // the gridctrl colours have been changed from the system colours.
- // If they have, then leave them, otherwise change them to reflect
- // the new system colours.
- m_crWindowText = ::GetSysColor(COLOR_WINDOWTEXT);
- m_crWindowColour = ::GetSysColor(COLOR_WINDOW);
- m_cr3DFace = ::GetSysColor(COLOR_3DFACE);
- m_crShadow = ::GetSysColor(COLOR_3DSHADOW);
- m_crGridColour = RGB(192,192,192);
- m_nRows = 0;
- m_nCols = 0;
- m_nFixedRows = 0;
- m_nFixedCols = 0;
- m_nDefCellHeight = 10; // These will get changed to something meaningful
- m_nDefCellWidth = 30; // when the window is created or subclassed
- m_nVScrollMax = 0; // Scroll position
- m_nHScrollMax = 0;
- m_nMargin = 0; // cell padding
- m_nRowsPerWheelNotch = GetMouseScrollLines(); // Get the number of lines
- // per mouse wheel notch to scroll
- m_MouseMode = MOUSE_NOTHING;
- m_nGridLines = GVL_BOTH;
- m_bEditable = TRUE;
- m_bListMode = FALSE;
- m_bSingleRowSelection = FALSE;
- m_bAllowDraw = TRUE; // allow draw updates
- m_bEnableSelection = TRUE;
- m_bAllowRowResize = TRUE;
- m_bAllowColumnResize = TRUE;
- m_bSortOnClick = TRUE; // Sort on header row click if in list mode
- m_bHandleTabKey = TRUE;
- #ifdef _WIN32_WCE
- m_bDoubleBuffer = FALSE; // Use double buffering to avoid flicker?
- #else
- m_bDoubleBuffer = TRUE; // Use double buffering to avoid flicker?
- #endif
- m_bTitleTips = TRUE; // show cell title tips
- m_bAscending = TRUE; // sorting stuff
- m_SortColumn = -1;
- m_nTimerID = 0; // For drag-selection
- m_nTimerInterval = 25; // (in milliseconds)
- m_nResizeCaptureRange = 3; // When resizing columns/row, the cursor has to be
- // within +/-3 pixels of the dividing line for
- // resizing to be possible
- m_pImageList = NULL;
- m_bAllowDragAndDrop = FALSE; // for drag and drop
- #ifndef _WIN32_WCE
- // Initially use the system message font for the GridCtrl font
- NONCLIENTMETRICS ncm;
- ncm.cbSize = sizeof(NONCLIENTMETRICS);
- VERIFY(SystemParametersInfo(SPI_GETNONCLIENTMETRICS, sizeof(NONCLIENTMETRICS), &ncm, 0));
- memcpy(&m_Logfont, &(ncm.lfMessageFont), sizeof(LOGFONT));
- #else
- LOGFONT lf;
- GetObject(GetStockObject(SYSTEM_FONT), sizeof(LOGFONT), &lf);
- memcpy(&m_Logfont, &lf, sizeof(LOGFONT));
- #endif
- // Set up the initial grid size
- SetRowCount(nRows);
- SetColumnCount(nCols);
- SetFixedRowCount(nFixedRows);
- SetFixedColumnCount(nFixedCols);
- // Set the colours
- SetTextColor(m_crWindowText);
- SetTextBkColor(m_crWindowColour);
- SetBkColor(m_crShadow);
- SetFixedTextColor(m_crWindowText);
- SetFixedBkColor(m_cr3DFace);
- // set initial selection range (ie. none)
- m_SelectedCellMap.RemoveAll();
- m_PrevSelectedCellMap.RemoveAll();
- }
- CGridCtrl::~CGridCtrl()
- {
- DeleteAllItems();
- DestroyWindow();
- m_Font.DeleteObject();
- #if !defined(GRIDCONTROL_NO_DRAGDROP) || !defined(GRIDCONTROL_NO_CLIPBOARD)
- // Uninitialize OLE support
- if (m_bMustUninitOLE)
- ::OleUninitialize();
- #endif
- }
- // Register the window class if it has not already been registered.
- BOOL CGridCtrl::RegisterWindowClass()
- {
- WNDCLASS wndcls;
- HINSTANCE hInst = AfxGetInstanceHandle();
- // HINSTANCE hInst = AfxGetResourceHandle();
- if (!(::GetClassInfo(hInst, GRIDCTRL_CLASSNAME, &wndcls)))
- {
- // otherwise we need to register a new class
- wndcls.style = CS_DBLCLKS | CS_HREDRAW | CS_VREDRAW;
- wndcls.lpfnWndProc = ::DefWindowProc;
- wndcls.cbClsExtra = wndcls.cbWndExtra = 0;
- wndcls.hInstance = hInst;
- wndcls.hIcon = NULL;
- wndcls.hCursor = AfxGetApp()->LoadStandardCursor(IDC_ARROW);
- wndcls.hbrBackground = (HBRUSH) (COLOR_3DFACE + 1);
- wndcls.lpszMenuName = NULL;
- wndcls.lpszClassName = GRIDCTRL_CLASSNAME;
- if (!AfxRegisterClass(&wndcls)) {
- AfxThrowResourceException();
- return FALSE;
- }
- }
- return TRUE;
- }
- BOOL CGridCtrl::Create(const RECT& rect, CWnd* pParentWnd, UINT nID, DWORD dwStyle)
- {
- ASSERT(pParentWnd->GetSafeHwnd());
- if (!CWnd::Create(GRIDCTRL_CLASSNAME, NULL, dwStyle, rect, pParentWnd, nID))
- return FALSE;
- #ifndef GRIDCONTROL_NO_DRAGDROP
- m_DropTarget.Register(this);
- #endif
- // Create titletips
- #ifndef GRIDCONTROL_NO_TITLETIPS
- if (m_bTitleTips)
- m_TitleTip.Create(this);
- #endif
- // The number of rows and columns will only be non-zero if the constructor
- // was called with non-zero initialising parameters. If this window was created
- // using a dialog template then the number of rows and columns will be 0 (which
- // means that the code below will not be needed - which is lucky 'cause it ain't
- // gonna get called in a dialog-template-type-situation.
- TRY {
- m_arRowHeights.SetSize(m_nRows); // initialize row heights
- m_arColWidths.SetSize(m_nCols); // initialize column widths
- }
- CATCH (CMemoryException, e) {
- e->ReportError();
- e->Delete();
- return FALSE;
- }
- END_CATCH
- for (int i = 0; i < m_nRows; i++) m_arRowHeights[i] = m_nDefCellHeight;
- for (int i = 0; i < m_nCols; i++) m_arColWidths[i] = m_nDefCellWidth;
- ResetScrollBars();
- return TRUE;
- }
- void CGridCtrl::PreSubclassWindow()
- {
- CWnd::PreSubclassWindow();
- HFONT hFont = ::CreateFontIndirect(&m_Logfont);
- OnSetFont((LPARAM)hFont, 0);
- DeleteObject(hFont);
-
- ResetScrollBars();
- }
- BOOL CGridCtrl::SubclassWindow(HWND hWnd)
- {
- if (!CWnd::SubclassWindow(hWnd))
- return FALSE;
- #ifndef GRIDCONTROL_NO_DRAGDROP
- m_DropTarget.Register(this);
- #endif
- #ifndef GRIDCONTROL_NO_TITLETIPS
- if (m_bTitleTips && !IsWindow(m_TitleTip.m_hWnd))
- m_TitleTip.Create(this);
- #endif
- return TRUE;
- }
- LRESULT CGridCtrl::SendMessageToParent(int nRow, int nCol, int nMessage)
- {
- if (!IsWindow(m_hWnd))
- return 0;
- NM_GRIDVIEW nmgv;
- nmgv.iRow = nRow;
- nmgv.iColumn = nCol;
- nmgv.hdr.hwndFrom = m_hWnd;
- nmgv.hdr.idFrom = GetDlgCtrlID();
- nmgv.hdr.code = nMessage;
- CWnd *pOwner = GetOwner();
- if (pOwner && IsWindow(pOwner->m_hWnd))
- return pOwner->SendMessage(WM_NOTIFY, nmgv.hdr.idFrom, (LPARAM)&nmgv);
- else
- return 0;
- }
- BEGIN_MESSAGE_MAP(CGridCtrl, CWnd)
- //{{AFX_MSG_MAP(CGridCtrl)
- ON_WM_PAINT()
- ON_WM_HSCROLL()
- ON_WM_VSCROLL()
- ON_WM_SIZE()
- ON_WM_LBUTTONUP()
- ON_WM_LBUTTONDOWN()
- ON_WM_MOUSEMOVE()
- ON_WM_TIMER()
- ON_WM_GETDLGCODE()
- ON_WM_KEYDOWN()
- ON_WM_CHAR()
- ON_WM_LBUTTONDBLCLK()
- ON_WM_ERASEBKGND()
- ON_COMMAND(ID_EDIT_SELECT_ALL, OnEditSelectAll)
- ON_UPDATE_COMMAND_UI(ID_EDIT_SELECT_ALL, OnUpdateEditSelectAll)
- //}}AFX_MSG_MAP
- #ifndef _WIN32_WCE_NO_CURSOR
- ON_WM_SETCURSOR()
- #endif
- #ifndef _WIN32_WCE
- ON_WM_SYSCOLORCHANGE()
- ON_WM_CAPTURECHANGED()
- #endif
- #ifndef GRIDCONTROL_NO_CLIPBOARD
- ON_COMMAND(ID_EDIT_COPY, OnEditCopy)
- ON_UPDATE_COMMAND_UI(ID_EDIT_COPY, OnUpdateEditCopy)
- ON_COMMAND(ID_EDIT_CUT, OnEditCut)
- ON_UPDATE_COMMAND_UI(ID_EDIT_CUT, OnUpdateEditCut)
- ON_COMMAND(ID_EDIT_PASTE, OnEditPaste)
- ON_UPDATE_COMMAND_UI(ID_EDIT_PASTE, OnUpdateEditPaste)
- #endif
- #if !defined(_WIN32_WCE) && (_MFC_VER >= 0x0421)
- ON_WM_MOUSEWHEEL()
- #endif
- #if (_WIN32_WCE >= 210)
- ON_WM_SETTINGCHANGE()
- #endif
- ON_MESSAGE(WM_SETFONT, OnSetFont)
- ON_MESSAGE(WM_GETFONT, OnGetFont)
- ON_NOTIFY(GVN_ENDLABELEDIT, IDC_INPLACE_CONTROL, OnEndInPlaceEdit)
- END_MESSAGE_MAP()
- /////////////////////////////////////////////////////////////////////////////
- // CGridCtrl message handlers
- void CGridCtrl::OnPaint()
- {
- CPaintDC dc(this); // device context for painting
- if (m_bDoubleBuffer) // Use a memory DC to remove flicker
- {
- CMyMemDC MemDC(&dc);
- OnDraw(&MemDC);
- }
- else // Draw raw - this helps in debugging vis problems.
- OnDraw(&dc);
- }
- BOOL CGridCtrl::OnEraseBkgnd(CDC* /*pDC*/)
- {
- return TRUE; // Don't erase the background.
- }
- // Custom background erasure. This gets called from within the OnDraw function,
- // since we will (most likely) be using a memory DC to stop flicker. If we just
- // erase the background normally through OnEraseBkgnd, and didn't fill the memDC's
- // selected bitmap with colour, then all sorts of vis problems would occur
- void CGridCtrl::EraseBkgnd(CDC* pDC)
- {
- CRect VisRect, ClipRect, rect;
- CBrush FixedBack(GetFixedBkColor()),
- TextBack(GetTextBkColor()),
- Back(GetBkColor());
- if (pDC->GetClipBox(ClipRect) == ERROR)
- return;
- GetVisibleNonFixedCellRange(VisRect);
- // Draw Fixed columns background
- int nFixedColumnWidth = GetFixedColumnWidth();
- if (ClipRect.left < nFixedColumnWidth && ClipRect.top < VisRect.bottom)
- pDC->FillRect(CRect(ClipRect.left, ClipRect.top,
- nFixedColumnWidth, VisRect.bottom),
- &FixedBack);
-
- // Draw Fixed rows background
- int nFixedRowHeight = GetFixedRowHeight();
- if (ClipRect.top < nFixedRowHeight &&
- ClipRect.right > nFixedColumnWidth && ClipRect.left < VisRect.right)
- pDC->FillRect(CRect(nFixedColumnWidth-1, ClipRect.top,
- VisRect.right, nFixedRowHeight),
- &FixedBack);
- // Draw non-fixed cell background
- if (rect.IntersectRect(VisRect, ClipRect))
- {
- CRect CellRect(max(nFixedColumnWidth, rect.left),
- max(nFixedRowHeight, rect.top),
- rect.right, rect.bottom);
- pDC->FillRect(CellRect, &TextBack);
- }
- // Draw right hand side of window outside grid
- if (VisRect.right < ClipRect.right)
- pDC->FillRect(CRect(VisRect.right, ClipRect.top,
- ClipRect.right, ClipRect.bottom),
- &Back);
- // Draw bottom of window below grid
- if (VisRect.bottom < ClipRect.bottom && ClipRect.left < VisRect.right)
- pDC->FillRect(CRect(ClipRect.left, VisRect.bottom,
- VisRect.right, ClipRect.bottom),
- &Back);
- }
- void CGridCtrl::OnSize(UINT nType, int cx, int cy)
- {
- static BOOL bAlreadyInsideThisProcedure = FALSE;
- if (bAlreadyInsideThisProcedure)
- return;
- if (!::IsWindow(m_hWnd))
- return;
- // Start re-entry blocking
- bAlreadyInsideThisProcedure = TRUE;
- // if (::IsWindow(GetSafeHwnd()) && GetFocus()->GetSafeHwnd() != GetSafeHwnd())
- SetFocus(); // Auto-destroy any InPlaceEdit's
- CWnd::OnSize(nType, cx, cy);
- ResetScrollBars();
- // End re-entry blocking
- bAlreadyInsideThisProcedure = FALSE;
- }
- UINT CGridCtrl::OnGetDlgCode()
- {
- UINT nCode = DLGC_WANTARROWS | DLGC_WANTCHARS; // DLGC_WANTALLKEYS; //
- if (m_bHandleTabKey && !IsCTRLpressed())
- nCode |= DLGC_WANTTAB;
- return nCode;
- }
- #ifndef _WIN32_WCE
- // If system colours change, then redo colours
- void CGridCtrl::OnSysColorChange()
- {
- CWnd::OnSysColorChange();
-
- if (GetTextColor() == m_crWindowText) // Still using system colours
- SetTextColor(::GetSysColor(COLOR_WINDOWTEXT)); // set to new system colour
- if (GetTextBkColor() == m_crWindowColour)
- SetTextBkColor(::GetSysColor(COLOR_WINDOW));
- if (GetBkColor() == m_crShadow)
- SetBkColor(::GetSysColor(COLOR_3DSHADOW));
- if (GetFixedTextColor() == m_crWindowText)
- SetFixedTextColor(::GetSysColor(COLOR_WINDOWTEXT));
- if (GetFixedBkColor() == m_cr3DFace)
- SetFixedBkColor(::GetSysColor(COLOR_3DFACE));
- m_crWindowText = ::GetSysColor(COLOR_WINDOWTEXT);
- m_crWindowColour = ::GetSysColor(COLOR_WINDOW);
- m_cr3DFace = ::GetSysColor(COLOR_3DFACE);
- m_crShadow = ::GetSysColor(COLOR_3DSHADOW);
- }
- #endif
- #ifndef _WIN32_WCE_NO_CURSOR
- // If we are drag-selecting cells, or drag and dropping, stop now
- void CGridCtrl::OnCaptureChanged(CWnd *pWnd)
- {
- if (pWnd->GetSafeHwnd() == GetSafeHwnd()) return;
- // kill timer if active
- if (m_nTimerID != 0)
- {
- KillTimer(m_nTimerID);
- m_nTimerID = 0;
- }
- #ifndef GRIDCONTROL_NO_DRAGDROP
- // Kill drag and drop if active
- if (m_MouseMode == MOUSE_DRAGGING)
- m_MouseMode = MOUSE_NOTHING;
- #endif
- }
- #endif
- #if (_MFC_VER >= 0x0421) || (_WIN32_WCE >= 210)
- // If system settings change, then redo colours
- void CGridCtrl::OnSettingChange(UINT uFlags, LPCTSTR lpszSection)
- {
- CWnd::OnSettingChange(uFlags, lpszSection);
-
- if (GetTextColor() == m_crWindowText) // Still using system colours
- SetTextColor(::GetSysColor(COLOR_WINDOWTEXT)); // set to new system colour
- if (GetTextBkColor() == m_crWindowColour)
- SetTextBkColor(::GetSysColor(COLOR_WINDOW));
- if (GetBkColor() == m_crShadow)
- SetBkColor(::GetSysColor(COLOR_3DSHADOW));
- if (GetFixedTextColor() == m_crWindowText)
- SetFixedTextColor(::GetSysColor(COLOR_WINDOWTEXT));
- if (GetFixedBkColor() == m_cr3DFace)
- SetFixedBkColor(::GetSysColor(COLOR_3DFACE));
- m_crWindowText = ::GetSysColor(COLOR_WINDOWTEXT);
- m_crWindowColour = ::GetSysColor(COLOR_WINDOW);
- m_cr3DFace = ::GetSysColor(COLOR_3DFACE);
- m_crShadow = ::GetSysColor(COLOR_3DSHADOW);
- m_nRowsPerWheelNotch = GetMouseScrollLines(); // Get the number of lines
- }
- #endif
- // For drag-selection. Scrolls hidden cells into view
- // TODO: decrease timer interval over time to speed up selection over time
- void CGridCtrl::OnTimer(UINT nIDEvent)
- {
- ASSERT(nIDEvent == WM_LBUTTONDOWN);
- if (nIDEvent != WM_LBUTTONDOWN)
- return;
- CPoint pt, origPt;
- #ifdef _WIN32_WCE
- if (m_MouseMode == MOUSE_NOTHING)
- return;
- origPt = GetMessagePos();
- #else
- if (!GetCursorPos(&origPt))
- return;
- #endif
- ScreenToClient(&origPt);
- CRect rect;
- GetClientRect(rect);
- int nFixedRowHeight = GetFixedRowHeight();
- int nFixedColWidth = GetFixedColumnWidth();
- pt = origPt;
- if (pt.y > rect.bottom)
- {
- //SendMessage(WM_VSCROLL, SB_LINEDOWN, 0);
- SendMessage(WM_KEYDOWN, VK_DOWN, 0);
- if (pt.x < rect.left)
- pt.x = rect.left;
- if (pt.x > rect.right)
- pt.x = rect.right;
- pt.y = rect.bottom;
- OnSelecting(GetCellFromPt(pt));
- }
- else if (pt.y < nFixedRowHeight)
- {
- //SendMessage(WM_VSCROLL, SB_LINEUP, 0);
- SendMessage(WM_KEYDOWN, VK_UP, 0);
- if (pt.x < rect.left)
- pt.x = rect.left;
- if (pt.x > rect.right)
- pt.x = rect.right;
- pt.y = nFixedRowHeight + 1;
- OnSelecting(GetCellFromPt(pt));
- }
- pt = origPt;
- if (pt.x > rect.right)
- {
- // SendMessage(WM_HSCROLL, SB_LINERIGHT, 0);
- SendMessage(WM_KEYDOWN, VK_RIGHT, 0);
- if (pt.y < rect.top)
- pt.y = rect.top;
- if (pt.y > rect.bottom)
- pt.y = rect.bottom;
- pt.x = rect.right;
- OnSelecting(GetCellFromPt(pt));
- }
- else if (pt.x < nFixedColWidth)
- {
- //SendMessage(WM_HSCROLL, SB_LINELEFT, 0);
- SendMessage(WM_KEYDOWN, VK_LEFT, 0);
- if (pt.y < rect.top)
- pt.y = rect.top;
- if (pt.y > rect.bottom)
- pt.y = rect.bottom;
- pt.x = nFixedColWidth + 1;
- OnSelecting(GetCellFromPt(pt));
- }
- }
- // move about with keyboard
- void CGridCtrl::OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags)
- {
- if (!IsValid(m_idCurrentCell))
- {
- CWnd::OnKeyDown(nChar, nRepCnt, nFlags);
- return;
- }
- CCellID next = m_idCurrentCell;
- BOOL bChangeLine = FALSE;
- if (IsCTRLpressed())
- {
- switch (nChar)
- {
- case 'A': OnEditSelectAll(); break;
- #ifndef GRIDCONTROL_NO_CLIPBOARD
- case 'X': OnEditCut(); break;
- case 'C': OnEditCopy(); break;
- case 'V': OnEditPaste(); break;
- #endif
- }
- }
- switch (nChar)
- {
- case VK_DELETE:
- if (IsCellEditable(m_idCurrentCell.row, m_idCurrentCell.col))
- {
- SendMessageToParent(m_idCurrentCell.row, m_idCurrentCell.col, GVN_BEGINLABELEDIT);
- SetItemText(m_idCurrentCell.row, m_idCurrentCell.col, _T(""));
- SetModified(TRUE, m_idCurrentCell.row, m_idCurrentCell.col);
- SendMessageToParent(m_idCurrentCell.row, m_idCurrentCell.col, GVN_ENDLABELEDIT);
- RedrawCell(m_idCurrentCell);
- }
- break;
- case VK_TAB:
- if (IsSHIFTpressed())
- {
- if (next.col > m_nFixedCols)
- next.col--;
- else if (next.col == m_nFixedCols && next.row > m_nFixedRows)
- {
- next.row--;
- next.col = GetColumnCount() - 1;
- bChangeLine = TRUE;
- }
- else
- CWnd::OnKeyDown(nChar, nRepCnt, nFlags);
- }
- else
- {
- if (next.col < (GetColumnCount() - 1))
- next.col++;
- else if (next.col == (GetColumnCount() - 1) &&
- next.row < (GetRowCount() - 1) )
- {
- next.row++;
- next.col = m_nFixedCols;
- bChangeLine = TRUE;
- }
- else
- CWnd::OnKeyDown(nChar, nRepCnt, nFlags);
- }
- break;
- case VK_DOWN:
- if (next.row < (GetRowCount() - 1))
- next.row++;
- break;
-
- case VK_UP:
- if (next.row > m_nFixedRows)
- next.row--;
- break;
- case VK_RIGHT:
- if (next.col < (GetColumnCount() - 1))
- next.col++;
- break;
-
- case VK_LEFT:
- if (next.col > m_nFixedCols)
- next.col--;
- break;
- case VK_NEXT:
- {
- CCellID idOldTopLeft = GetTopleftNonFixedCell();
- SendMessage(WM_VSCROLL, SB_PAGEDOWN, 0);
- CCellID idNewTopLeft = GetTopleftNonFixedCell();
- int increment = idNewTopLeft.row - idOldTopLeft.row;
- if (increment) {
- next.row += increment;
- if (next.row > (GetRowCount() - 1))
- next.row = GetRowCount() - 1;
- }
- else
- next.row = GetRowCount() - 1;
- break;
- }
-
- case VK_PRIOR:
- {
- CCellID idOldTopLeft = GetTopleftNonFixedCell();
- SendMessage(WM_VSCROLL, SB_PAGEUP, 0);
- CCellID idNewTopLeft = GetTopleftNonFixedCell();
- int increment = idNewTopLeft.row - idOldTopLeft.row;
- if (increment)
- {
- next.row += increment;
- if (next.row < m_nFixedRows)
- next.row = m_nFixedRows;
- } else
- next.row = m_nFixedRows;
- break;
- }
-
- case VK_HOME:
- SendMessage(WM_VSCROLL, SB_TOP, 0);
- next.row = m_nFixedRows;
- break;
-
- case VK_END:
- SendMessage(WM_VSCROLL, SB_BOTTOM, 0);
- next.row = GetRowCount() - 1;
- break;
-
- default:
- CWnd::OnKeyDown(nChar, nRepCnt, nFlags);
- }
-
- if (next != m_idCurrentCell)
- {
- // While moving with the Cursorkeys the current ROW/CELL will get selected
- // OR Selection will get expanded when SHIFT is pressed
- // Cut n paste from OnLButtonDown - Franco Bez
- // Added check for NULL mouse mode - Chris Maunder.
- if (m_MouseMode == MOUSE_NOTHING)
- {
- m_PrevSelectedCellMap.RemoveAll();
- m_MouseMode = m_bListMode? MOUSE_SELECT_ROW : MOUSE_SELECT_CELLS;
- if (!IsSHIFTpressed() || nChar == VK_TAB)
- m_SelectionStartCell = next;
- OnSelecting(next);
- m_MouseMode = MOUSE_NOTHING;
- }
- SetFocusCell(next);
- if (!IsCellVisible(next))
- {
- EnsureVisible(next); // Make sure cell is visible
- switch (nChar) {
- case VK_RIGHT:
- SendMessage(WM_HSCROLL, SB_LINERIGHT, 0);
- break;
- case VK_LEFT:
- SendMessage(WM_HSCROLL, SB_LINELEFT, 0);
- break;
- case VK_DOWN:
- SendMessage(WM_VSCROLL, SB_LINEDOWN, 0);
- break;
-
- case VK_UP:
- SendMessage(WM_VSCROLL, SB_LINEUP, 0);
- break;
-
- case VK_TAB:
- if (IsSHIFTpressed())
- {
- if (bChangeLine)
- {
- SendMessage(WM_VSCROLL, SB_LINEUP, 0);
- SetScrollPos32(SB_HORZ, m_nHScrollMax);
- break;
- }
- else
- SendMessage(WM_HSCROLL, SB_LINELEFT, 0);
- }
- else
- {
- if (bChangeLine)
- {
- SendMessage(WM_VSCROLL, SB_LINEDOWN, 0);
- SetScrollPos32(SB_HORZ, 0);
- break;
- }
- else
- SendMessage(WM_HSCROLL, SB_LINERIGHT, 0);
- }
- break;
- }
- Invalidate();
- }
- }
- }
- // Instant editing of cells when keys are pressed
- void CGridCtrl::OnChar(UINT nChar, UINT nRepCnt, UINT nFlags)
- {
- if (!IsCTRLpressed() && m_MouseMode == MOUSE_NOTHING)
- {
- if (!m_bHandleTabKey || (m_bHandleTabKey && nChar != VK_TAB))
- OnEditCell(m_idCurrentCell.row, m_idCurrentCell.col, nChar);
- }
- CWnd::OnChar(nChar, nRepCnt, nFlags);
- }
- // Callback from any CInPlaceEdits that ended. This just calls OnEndEditCell,
- // refreshes the edited cell and moves onto next cell if the return character
- // from the edit says we should.
- void CGridCtrl::OnEndInPlaceEdit(NMHDR* pNMHDR, LRESULT* pResult)
- {
- GV_DISPINFO *pgvDispInfo = (GV_DISPINFO *)pNMHDR;
- GV_ITEM *pgvItem = &pgvDispInfo->item;
- // In case OnEndInPlaceEdit called as window is being destroyed
- if (!IsWindow(GetSafeHwnd()))
- return;
- OnEndEditCell(pgvItem->row, pgvItem->col, pgvItem->szText);
- InvalidateCellRect(CCellID(pgvItem->row, pgvItem->col));
- SendMessageToParent(pgvItem->row, pgvItem->col, GVN_ENDLABELEDIT);
- switch (pgvItem->lParam)
- {
- case VK_TAB:
- case VK_DOWN:
- case VK_UP:
- case VK_RIGHT:
- case VK_LEFT:
- case VK_NEXT:
- case VK_PRIOR:
- case VK_HOME:
- case VK_END:
- OnKeyDown(pgvItem->lParam, 0, 0);
- OnEditCell(m_idCurrentCell.row, m_idCurrentCell.col, pgvItem->lParam);
- }
- *pResult = 0;
- }
- // Handle horz scrollbar notifications
- void CGridCtrl::OnHScroll(UINT nSBCode, UINT /*nPos*/, CScrollBar* /*pScrollBar*/)
- {
- if (GetFocus()->GetSafeHwnd() != GetSafeHwnd())
- SetFocus(); // Auto-destroy any InPlaceEdit's
- #ifndef GRIDCONTROL_NO_TITLETIPS
- // m_TitleTip.Hide(); // hide any titletips
- #endif
- int scrollPos = GetScrollPos32(SB_HORZ);
- CCellID idTopLeft = GetTopleftNonFixedCell();
- CRect rect;
- GetClientRect(rect);
- switch (nSBCode)
- {
- case SB_LINERIGHT:
- if (scrollPos < m_nHScrollMax)
- {
- int xScroll = GetColumnWidth(idTopLeft.col);
- SetScrollPos32(SB_HORZ, scrollPos + xScroll);
- if (GetScrollPos32(SB_HORZ) == scrollPos) break; // didn't work
- rect.left = GetFixedColumnWidth() + xScroll;
- ScrollWindow(-xScroll, 0, rect);
- rect.left = rect.right - xScroll;
- InvalidateRect(rect);
- }
- break;
- case SB_LINELEFT:
- if (scrollPos > 0 && idTopLeft.col > GetFixedColumnCount())
- {
- int xScroll = GetColumnWidth(idTopLeft.col-1);
- SetScrollPos32(SB_HORZ, max(0,scrollPos - xScroll));
- rect.left = GetFixedColumnWidth();
- ScrollWindow(xScroll, 0, rect);
- rect.right = rect.left + xScroll;
- InvalidateRect(rect);
- }
- break;
- case SB_PAGERIGHT:
- if (scrollPos < m_nHScrollMax)
- {
- rect.left = GetFixedColumnWidth();
- int offset = rect.Width();
- int pos = min(m_nHScrollMax, scrollPos + offset);
- SetScrollPos32(SB_HORZ, pos);
- rect.left = GetFixedColumnWidth();
- InvalidateRect(rect);
- }
- break;
- case SB_PAGELEFT:
- if (scrollPos > 0)
- {
- rect.left = GetFixedColumnWidth();
- int offset = -rect.Width();
- int pos = max(0, scrollPos + offset);
- SetScrollPos32(SB_HORZ, pos);
- rect.left = GetFixedColumnWidth();
- InvalidateRect(rect);
- }
- break;
- case SB_THUMBPOSITION:
- case SB_THUMBTRACK:
- {
- SetScrollPos32(SB_HORZ, GetScrollPos32(SB_HORZ, TRUE));
- CCellID idNewTopLeft = GetTopleftNonFixedCell();
- if (idNewTopLeft != idTopLeft)
- {
- rect.left = GetFixedColumnWidth();
- InvalidateRect(rect);
- }
- }
- break;
- case SB_LEFT:
- if (scrollPos > 0)
- {
- SetScrollPos32(SB_HORZ, 0);
- Invalidate();
- }
- break;
- case SB_RIGHT:
- if (scrollPos < m_nHScrollMax)
- {
- SetScrollPos32(SB_HORZ, m_nHScrollMax);
- Invalidate();
- }
- break;
- default: break;
- }
- }
- // Handle vert scrollbar notifications
- void CGridCtrl::OnVScroll(UINT nSBCode, UINT /*nPos*/, CScrollBar* /*pScrollBar*/)
- {
- if (GetFocus()->GetSafeHwnd() != GetSafeHwnd())
- SetFocus(); // Auto-destroy any InPlaceEdit's
- #ifndef GRIDCONTROL_NO_TITLETIPS
- // m_TitleTip.Hide(); // hide any titletips
- #endif
- // Get the scroll position ourselves to ensure we get a 32 bit value
- int scrollPos = GetScrollPos32(SB_VERT);
- CCellID idTopLeft = GetTopleftNonFixedCell();
- CRect rect;
- GetClientRect(rect);
- switch (nSBCode)
- {
- case SB_LINEDOWN:
- if (scrollPos < m_nVScrollMax)
- {
- int yScroll = GetRowHeight(idTopLeft.row);
- SetScrollPos32(SB_VERT, scrollPos + yScroll);
- if (GetScrollPos32(SB_VERT) == scrollPos)
- break; // didn't work
- rect.top = GetFixedRowHeight() + yScroll;
- ScrollWindow( 0, -yScroll, rect);
- rect.top = rect.bottom - yScroll;
- InvalidateRect(rect);
- }
- break;
- case SB_LINEUP:
- if (scrollPos > 0 && idTopLeft.row > GetFixedRowCount())
- {
- int yScroll = GetRowHeight(idTopLeft.row-1);
- SetScrollPos32(SB_VERT, max(0, scrollPos - yScroll));
- rect.top = GetFixedRowHeight();
- ScrollWindow(0, yScroll, rect);
- rect.bottom = rect.top + yScroll;
- InvalidateRect(rect);
- }
- break;
- case SB_PAGEDOWN:
- if (scrollPos < m_nVScrollMax)
- {
- rect.top = GetFixedRowHeight();
- scrollPos = min(m_nVScrollMax, scrollPos + rect.Height());
- SetScrollPos32(SB_VERT, scrollPos);
- rect.top = GetFixedRowHeight();
- InvalidateRect(rect);
- }
- break;
- case SB_PAGEUP:
- if (scrollPos > 0)
- {
- rect.top = GetFixedRowHeight();
- int offset = -rect.Height();
- int pos = max(0, scrollPos + offset);
- SetScrollPos32(SB_VERT, pos);
- rect.top = GetFixedRowHeight();
- InvalidateRect(rect);
- }
- break;
- case SB_THUMBPOSITION:
- case SB_THUMBTRACK:
- {
- SetScrollPos32(SB_VERT, GetScrollPos32(SB_VERT, TRUE));
- CCellID idNewTopLeft = GetTopleftNonFixedCell();
- if (idNewTopLeft != idTopLeft)
- {
- rect.top = GetFixedRowHeight();
- InvalidateRect(rect);
- }
- }
- break;
- case SB_TOP:
- if (scrollPos > 0)
- {
- SetScrollPos32(SB_VERT, 0);
- Invalidate();
- }
- break;
- case SB_BOTTOM:
- if (scrollPos < m_nVScrollMax)
- {
- SetScrollPos32(SB_VERT, m_nVScrollMax);
- Invalidate();
- }
- default: break;
- }
- }
- /////////////////////////////////////////////////////////////////////////////
- // CGridCtrl implementation functions
- void CGridCtrl::OnDraw(CDC* pDC)
- {
- CRect rect;
- int row,col;
- CRect clipRect;
- if (pDC->GetClipBox(&clipRect) == ERROR)
- return;
- EraseBkgnd(pDC); // OnEraseBkgnd does nothing, so erase bkgnd here.
- // This necessary since we may be using a Memory DC.
- int nFixedRowHeight = GetFixedRowHeight();
- int nFixedColWidth = GetFixedColumnWidth();
- CCellID idTopLeft = GetTopleftNonFixedCell();
- int minVisibleRow = idTopLeft.row,
- minVisibleCol = idTopLeft.col;
- CRect VisRect;
- CCellRange VisCellRange = GetVisibleNonFixedCellRange(VisRect);
- int maxVisibleRow = VisCellRange.GetMaxRow(),
- maxVisibleCol = VisCellRange.GetMaxCol();
- // draw top-left cells 0..m_nFixedRows-1, 0..m_nFixedCols-1
- rect.bottom = -1;
- for (row = 0; row < m_nFixedRows; row++)
- {
- rect.top = rect.bottom+1;
- rect.bottom = rect.top + GetRowHeight(row)-1;
- rect.right = -1;
- for (col = 0; col < m_nFixedCols; col++)
- {
- rect.left = rect.right+1;
- rect.right = rect.left + GetColumnWidth(col)-1;
-
- DrawFixedCell(pDC, row, col, rect);
- }
- }
-
- // draw fixed column cells: m_nFixedRows..n, 0..m_nFixedCols-1
- rect.bottom = nFixedRowHeight-1;
- for (row = minVisibleRow; row <= maxVisibleRow; row++)
- {
- rect.top = rect.bottom+1;
- rect.bottom = rect.top + GetRowHeight(row)-1;
- // rect.bottom = bottom pixel of previous row
- if (rect.top > clipRect.bottom)
- break; // Gone past cliprect
- if (rect.bottom < clipRect.top)
- continue; // Reached cliprect yet?
- rect.right = -1;
- for (col = 0; col < m_nFixedCols; col++)
- {
- rect.left = rect.right+1;
- rect.right = rect.left + GetColumnWidth(col)-1;
- if (rect.left > clipRect.right)
- break; // gone past cliprect
- if (rect.right < clipRect.left)
- continue; // Reached cliprect yet?
- DrawFixedCell(pDC, row, col, rect);
- }
- }
-
- // draw fixed row cells 0..m_nFixedRows, m_nFixedCols..n
- rect.bottom = -1;
- for (row = 0; row < m_nFixedRows; row++)
- {
- rect.top = rect.bottom+1;
- rect.bottom = rect.top + GetRowHeight(row)-1;
- // rect.bottom = bottom pixel of previous row
- if (rect.top > clipRect.bottom)
- break; // Gone past cliprect
- if (rect.bottom < clipRect.top)
- continue; // Reached cliprect yet?
- rect.right = nFixedColWidth-1;
- for (col = minVisibleCol; col <= maxVisibleCol; col++)
- {
- rect.left = rect.right+1;
- rect.right = rect.left + GetColumnWidth(col)-1;
- if (rect.left > clipRect.right)
- break; // gone past cliprect
- if (rect.right < clipRect.left)
- continue; // Reached cliprect yet?
- DrawFixedCell(pDC, row, col, rect);
- }
- }
-
- // draw rest of non-fixed cells
- rect.bottom = nFixedRowHeight-1;
- for (row = minVisibleRow; row <= maxVisibleRow; row++)
- {
- rect.top = rect.bottom+1;
- rect.bottom = rect.top + GetRowHeight(row)-1;
- // rect.bottom = bottom pixel of previous row
- if (rect.top > clipRect.bottom)
- break; // Gone past cliprect
- if (rect.bottom < clipRect.top)
- continue; // Reached cliprect yet?
- rect.right = nFixedColWidth-1;
- for (col = minVisibleCol; col <= maxVisibleCol; col++)
- {
- rect.left = rect.right+1;
- rect.right = rect.left + GetColumnWidth(col)-1;
- if (rect.left > clipRect.right)
- break; // gone past cliprect
- if (rect.right < clipRect.left)
- continue; // Reached cliprect yet?
- DrawCell(pDC, row, col, rect);
- }
- }
- CPen pen;
- TRY {
- pen.CreatePen(PS_SOLID, 0, m_crGridColour);
- }
- CATCH (CResourceException, e)
- {
- e->Delete();
- return;
- }
- END_CATCH
- pDC->SelectObject(&pen);
- // draw vertical lines (drawn at ends of cells)
- if (m_nGridLines == GVL_BOTH || m_nGridLines == GVL_VERT)
- {
- int x = nFixedColWidth;
- for (col = minVisibleCol; col <= maxVisibleCol; col++) {
- x += GetColumnWidth(col);
- pDC->MoveTo(x-1, nFixedRowHeight);
- pDC->LineTo(x-1, VisRect.bottom);
- }
- }
-
- // draw horizontal lines (drawn at bottom of each cell)
- if (m_nGridLines == GVL_BOTH || m_nGridLines == GVL_HORZ)
- {
- int y = nFixedRowHeight;
- for (row = minVisibleRow; row <= maxVisibleRow; row++) {
- y += GetRowHeight(row);
- pDC->MoveTo(nFixedColWidth, y-1);
- pDC->LineTo(VisRect.right, y-1);
- }
- }
- pDC->SelectStockObject(NULL_PEN);
- #ifdef USE_MEMDC // Use a memDC for flicker free update
- }
- #else // Use normal DC - this helps in debugging
- }
- #endif
- ////////////////////////////////////////////////////////////////////////////////////////
- // CGridCtrl Cell selection stuff
- BOOL CGridCtrl::IsValid(int nRow, int nCol) const
- {
- return (nRow >= 0 && nRow < m_nRows && nCol >= 0 && nCol < m_nCols);
- }
- BOOL CGridCtrl::IsValid(const CCellID& cell) const
- {
- return IsValid(cell.row, cell.col);
- }
- BOOL CGridCtrl::IsValid(const CCellRange& range) const
- {
- return (range.GetMinRow() >= 0 && range.GetMinCol() >= 0 &&
- range.GetMaxRow() >= 0 && range.GetMaxCol() >= 0 &&
- range.GetMaxRow() < m_nRows && range.GetMaxCol() < m_nCols &&
- range.GetMinRow() <= range.GetMaxRow() && range.GetMinCol() <= range.GetMaxCol());
- }
- // Enables/Disables redraw for certain operations like columns auto-sizing etc,
- // but not for user caused things such as selection changes.
- void CGridCtrl::SetRedraw(BOOL bAllowDraw, BOOL bResetScrollBars /* = FALSE */)
- {
- TRACE(_T("%s: Setting redraw to %s\n"),
- GetRuntimeClass()->m_lpszClassName, bAllowDraw? _T("TRUE") : _T("FALSE"));
- if (bAllowDraw && !m_bAllowDraw)
- Invalidate();
- m_bAllowDraw = bAllowDraw;
- if (bResetScrollBars)
- ResetScrollBars();
- }
- // Forces a redraw of a cell immediately (using a direct DC construction,
- // or the supplied dc)
- BOOL CGridCtrl::RedrawCell(const CCellID& cell, CDC* pDC /* = NULL */)
- {
- return RedrawCell(cell.row, cell.col, pDC);
- }
- BOOL CGridCtrl::RedrawCell(int nRow, int nCol, CDC* pDC /* = NULL */)
- {
- BOOL bResult = TRUE;
- BOOL bMustReleaseDC = FALSE;
- if (!m_bAllowDraw || !IsCellVisible(nRow, nCol))
- return FALSE;
- CRect rect;
- if (!GetCellRect(nRow, nCol, rect))
- return FALSE;
- if (!pDC) {
- pDC = GetDC();
- if (pDC)
- bMustReleaseDC = TRUE;
- }
- if (pDC)
- {
- // Redraw cells directly
- if (nRow < m_nFixedRows || nCol < m_nFixedCols)
- {
- bResult = DrawFixedCell(pDC, nRow, nCol, rect, TRUE);
- }
- else
- {
- bResult = DrawCell(pDC, nRow, nCol, rect, TRUE);
-
- // Since we have erased the background, we will need to redraw the gridlines
- CPen pen;
- TRY {
- pen.CreatePen(PS_SOLID, 0, m_crGridColour);
- } CATCH (CException, e) {e->Delete();} END_CATCH
-
- CPen* pOldPen = (CPen*) pDC->SelectObject(&pen);
- if (m_nGridLines == GVL_BOTH || m_nGridLines == GVL_HORZ)
- {
- pDC->MoveTo(rect.left, rect.bottom);
- pDC->LineTo(rect.right+1, rect.bottom);
- }
- if (m_nGridLines == GVL_BOTH || m_nGridLines == GVL_VERT)
- {
- pDC->MoveTo(rect.right, rect.top);
- pDC->LineTo(rect.right, rect.bottom+1);
- }
- pDC->SelectObject(pOldPen);
- }
- } else
- InvalidateRect(rect, TRUE); // Could not get a DC - invalidate it anyway
- // and hope that OnPaint manages to get one
- if (bMustReleaseDC)
- ReleaseDC(pDC);
- return bResult;
- }
- // redraw a complete row
- BOOL CGridCtrl::RedrawRow(int row)
- {
- BOOL bResult = TRUE;
- CDC* pDC = GetDC();
- for (int col = 0; col < GetColumnCount(); col++)
- bResult = RedrawCell(row, col, pDC) && bResult;
- if (pDC)
- ReleaseDC(pDC);
- return bResult;
- }
- // redraw a complete column
- BOOL CGridCtrl::RedrawColumn(int col)
- {
- BOOL bResult = TRUE;
- CDC* pDC = GetDC();
- for (int row = 0; row < GetRowCount(); row++)
- bResult = RedrawCell(row, col, pDC) && bResult;
- if (pDC)
- ReleaseDC(pDC);
- return bResult;
- }
- // Sets the currently selected cell, returning the previous current cell
- CCellID CGridCtrl::SetFocusCell(int nRow, int nCol)
- {
- return SetFocusCell(CCellID(nRow, nCol));
- }
- CCellID CGridCtrl::SetFocusCell(CCellID cell)
- {
- if (cell == m_idCurrentCell)
- return m_idCurrentCell;
- CCellID idPrev = m_idCurrentCell;
- m_idCurrentCell = cell;
- if (IsValid(idPrev))
- {
- SendMessageToParent(idPrev.row, idPrev.col, GVN_SELCHANGING);
- SetItemState(idPrev.row, idPrev.col,
- GetItemState(idPrev.row, idPrev.col) & ~GVIS_FOCUSED);
- RedrawCell(idPrev);
- if (idPrev.col != m_idCurrentCell.col)
- for (int row = 0; row < m_nFixedRows; row++)
- RedrawCell(row, idPrev.col);
- if (idPrev.row != m_idCurrentCell.row)
- for (int col = 0; col < m_nFixedCols; col++)
- RedrawCell(idPrev.row, col);
- }
- if (IsValid(m_idCurrentCell)) {
- SetItemState(m_idCurrentCell.row, m_idCurrentCell.col,
- GetItemState(m_idCurrentCell.row, m_idCurrentCell.col) | GVIS_FOCUSED);
- RedrawCell(m_idCurrentCell);
- if (idPrev.col != m_idCurrentCell.col)
- for (int row = 0; row < m_nFixedRows; row++)
- RedrawCell(row, m_idCurrentCell.col);
- if (idPrev.row != m_idCurrentCell.row)
- for (int col = 0; col < m_nFixedCols; col++)
- RedrawCell(m_idCurrentCell.row, col);
- SendMessageToParent(m_idCurrentCell.row, m_idCurrentCell.col, GVN_SELCHANGED);
- }
- return idPrev;
- }
- // Sets the range of currently selected cells
- void CGridCtrl::SetSelectedRange(const CCellRange& Range,
- BOOL bForceRepaint /* = FALSE */)
- {
- SetSelectedRange(Range.GetMinRow(), Range.GetMinCol(),
- Range.GetMaxRow(), Range.GetMaxCol(),
- bForceRepaint);
- }
- void CGridCtrl::SetSelectedRange(int nMinRow, int nMinCol, int nMaxRow, int nMaxCol,
- BOOL bForceRepaint /* = FALSE */)
- {
- if (!m_bEnableSelection) return;
- CDC* pDC = NULL;
- if (bForceRepaint) pDC = GetDC();
- // Unselect all previously selected cells
- for (POSITION pos = m_SelectedCellMap.GetStartPosition(); pos != NULL; )
- {
- DWORD key;
- CCellID cell;
- m_SelectedCellMap.GetNextAssoc(pos, key, (CCellID&)cell);
- // Reset the selection flag on the cell
- if (IsValid(cell)) {
- SetItemState(cell.row, cell.col,
- GetItemState(cell.row, cell.col) & ~GVIS_SELECTED);
- // If this is to be reselected, continue on past the redraw
- if (nMinRow <= cell.row && cell.row <= nMaxRow &&
- nMinCol <= cell.col && cell.col <= nMaxCol)
- continue;
- if (bForceRepaint && pDC) // Redraw NOW
- RedrawCell(cell.row, cell.col, pDC);
- else
- InvalidateCellRect(cell); // Redraw at leisure
- }
- }
-
- // if any previous selected cells are to be retained (eg Ctrl is being held down)
- // then copy them to the newly created list, and mark all these cells as
- // selected
- if (!GetSingleRowSelection())
- for (POSITION pos = m_PrevSelectedCellMap.GetStartPosition(); pos != NULL; /* nothing */)
- {
- DWORD key;
- CCellID cell;
- m_PrevSelectedCellMap.GetNextAssoc(pos, key, (CCellID&)cell);
- if (!IsValid(cell))
- continue;
- int nState = GetItemState(cell.row, cell.col);
- // Set state as Selected. This will add the cell to m_SelectedCells[]
- SetItemState(cell.row, cell.col, nState | GVIS_SELECTED);
- // Redraw (immediately or at leisure)
- if (bForceRepaint && pDC)
- RedrawCell(cell.row, cell.col, pDC);
- else
- InvalidateCellRect(cell);
- }
- // Now select all cells in the cell range specified. If the cell has already
- // been marked as selected (above) then ignore it.
- if (nMinRow >= 0 && nMinCol >= 0 && nMaxRow >= 0 && nMaxCol >= 0 &&
- nMaxRow < m_nRows && nMaxCol < m_nCols &&
- nMinRow <= nMaxRow && nMinCol <= nMaxCol)
- {
- for (int row = nMinRow; row <= nMaxRow; row++)
- for (int col = nMinCol; col <= nMaxCol; col++)
- {
- int nState = GetItemState(row, col);
- if (nState & GVIS_SELECTED)
- continue; // Already selected - ignore
- // Add to list of selected cells
- CCellID cell(row, col);
- // Set state as Selected. This will add the cell to m_SelectedCells[]
- SetItemState(row, col, nState | GVIS_SELECTED);
- // Redraw (immediately or at leisure)
- if (bForceRepaint && pDC)
- RedrawCell(row, col, pDC);
- else
- InvalidateCellRect(cell);
- }
- }
- // TRACE(_T("%d cells selected.\n"), m_SelectedCellMap.GetCount());
- if (pDC != NULL)
- ReleaseDC(pDC);
- }
- // selects all cells
- void CGridCtrl::SelectAllCells()
- {
- if (!m_bEnableSelection)
- return;
- SetSelectedRange(m_nFixedRows, m_nFixedCols, GetRowCount()-1, GetColumnCount()-1);
- }
- // selects columns
- void CGridCtrl::SelectColumns(CCellID currentCell)
- {
- if (!m_bEnableSelection)
- return;
- //if (currentCell.col == m_idCurrentCell.col) return;
- if (currentCell.col < m_nFixedCols)
- return;
- if (!IsValid(currentCell))
- return;
- SetSelectedRange(GetFixedRowCount(),
- min(m_SelectionStartCell.col, currentCell.col),
- GetRowCount()-1,
- max(m_SelectionStartCell.col, currentCell.col));
- }
- // selects rows
- void CGridCtrl::SelectRows(CCellID currentCell)
- {
- if (!m_bEnableSelection)
- return;
- //if (currentCell.row; == m_idCurrentCell.row) return;
- if (currentCell.row < m_nFixedRows)
- return;
- if (!IsValid(currentCell))
- return;
- if (GetListMode() && GetSingleRowSelection())
- SetSelectedRange(currentCell.row, GetFixedColumnCount(),
- currentCell.row, GetColumnCount()-1);
- else
- SetSelectedRange(min(m_SelectionStartCell.row, currentCell.row),
- GetFixedColumnCount(),
- max(m_SelectionStartCell.row, currentCell.row),
- GetColumnCount()-1);
- }
- // selects cells
- void CGridCtrl::SelectCells(CCellID currentCell)
- {
- if (!m_bEnableSelection)
- return;
- int row = currentCell.row;
- int col = currentCell.col;
- if (row < m_nFixedRows || col < m_nFixedCols)
- return;
- if (!IsValid(currentCell))
- return;
- // Prevent unnecessary redraws
- //if (currentCell == m_LeftClickDownCell) return;
- //else if (currentCell == m_idCurrentCell) return;
- SetSelectedRange(min(m_SelectionStartCell.row, row),
- min(m_SelectionStartCell.col, col),
- max(m_SelectionStartCell.row, row),
- max(m_SelectionStartCell.col, col));
- }
- void CGridCtrl::OnSelecting(const CCellID& currentCell)
- {
- if (!m_bEnableSelection)
- return;
- switch(m_MouseMode)
- {
- case MOUSE_SELECT_ALL: SelectAllCells(); break;
- case MOUSE_SELECT_COL: SelectColumns(currentCell); break;
- case MOUSE_SELECT_ROW: SelectRows(currentCell); break;
- case MOUSE_SELECT_CELLS: SelectCells(currentCell); break;
- }
- }
- #ifndef GRIDCONTROL_NO_CLIPBOARD
- ////////////////////////////////////////////////////////////////////////////////////////
- // Clipboard functions
- void CGridCtrl::CutSelectedText()
- {
- if (!IsEditable())
- return;
- // Clear contents of selected cells.
- for (POSITION pos = m_SelectedCellMap.GetStartPosition(); pos != NULL; )
- {
- DWORD key;
- CCellID cell;
- m_SelectedCellMap.GetNextAssoc(pos, key, (CCellID&)cell);
- if (!IsCellEditable(cell))
- continue;
- CGridCell* pCell = GetCell(cell.row, cell.col);
- if (pCell)
- {
- SendMessageToParent(cell.row, cell.col, GVN_BEGINLABELEDIT);
- EmptyCell(pCell, cell.row, cell.col);
- SetModified(TRUE, cell.row, cell.col);
- SendMessageToParent(cell.row, cell.col, GVN_ENDLABELEDIT);
- }
- }
- Invalidate();
- }
- COleDataSource* CGridCtrl::CopyTextFromGrid()
- {
- USES_CONVERSION;
- CCellRange Selection = GetSelectedCellRange();
- if (!IsValid(Selection)) return NULL;
- // Write to shared file (REMEBER: CF_TEXT is ANSI, not UNICODE, so we need to convert)
- CSharedFile sf(GMEM_MOVEABLE|GMEM_DDESHARE|GMEM_ZEROINIT);
- // Get a tab delimited string to copy to cache
- CString str;
- CGridCell *pCell;
- for (int row = Selection.GetMinRow(); row <= Selection.GetMaxRow(); row++)
- {
- str.Empty();
- for (int col = Selection.GetMinCol(); col <= Selection.GetMaxCol(); col++)
- {
- pCell = GetCell(row,col);
- if (pCell && (pCell->state & GVIS_SELECTED))
- {
- if (pCell->szText.IsEmpty())
- str += _T(" ");
- else
- str += pCell->szText;
- }
- if (col != Selection.GetMaxCol())
- str += _T("\t");
- }
- if (row != Selection.GetMaxRow())
- str += _T("\n");
- sf.Write(T2A(str.GetBuffer(1)), str.GetLength());
- str.ReleaseBuffer();
- }
- char c = '\0';
- sf.Write(&c, 1);
- DWORD dwLen = (DWORD)sf.GetLength();
- HGLOBAL hMem = sf.Detach();
- if (!hMem)
- return NULL;
- hMem = ::GlobalReAlloc(hMem, dwLen, GMEM_MOVEABLE|GMEM_DDESHARE|GMEM_ZEROINIT);
- if (!hMem)
- return NULL;
- // Cache data
- COleDataSource* pSource = new COleDataSource();
- pSource->CacheGlobalData(CF_TEXT, hMem);
- return pSource;
- }
- BOOL CGridCtrl::PasteTextToGrid(CCellID cell, COleDataObject* pDataObject)
- {
- if (!IsValid(cell) || !IsCellEditable(cell) || !pDataObject->IsDataAvailable(CF_TEXT))
- return FALSE;
- // Get the text from the COleDataObject
- HGLOBAL hmem = pDataObject->GetGlobalData(CF_TEXT);
- CMemFile sf((BYTE*) ::GlobalLock(hmem), ::GlobalSize(hmem));
- // CF_TEXT is ANSI text, so we need to allocate a char* buffer
- // to hold this.
- LPSTR szBuffer = new char[::GlobalSize(hmem)];
- if (!szBuffer)
- return FALSE;
- sf.Read(szBuffer, ::GlobalSize(hmem));
- ::GlobalUnlock(hmem);
- // Now store in generic TCHAR form so we no longer have to deal with
- // ANSI/UNICODE problems
- CString strText = szBuffer;
- delete szBuffer;
- // Parse text data and set in cells...
- strText.LockBuffer();
- CString strLine = strText;
- int nLine = 0;
- // Find the end of the first line
- int nIndex;
- do {
- int nColumn = 0;
- nIndex = strLine.Find(_T("\n"));
- // Store the remaining chars after the newline
- CString strNext = (nIndex < 0)? _T("") : strLine.Mid(nIndex+1);
- // Remove all chars after the newline
- if (nIndex >= 0)
- strLine = strLine.Left(nIndex);
- // Make blank entries a "space"
- if (strLine.IsEmpty() && nIndex >= 0)
- strLine = _T(" ");
- LPTSTR szLine = strLine.GetBuffer(1);
- // Break the current line into tokens (tab or comma delimited)
- LPTSTR pszCellText = _tcstok(szLine, _T("\t,\n"));
- while (pszCellText != NULL)
- {
- CCellID TargetCell(cell.row + nLine, cell.col + nColumn);
- if (IsValid(TargetCell))
- {
- CString strCellText = pszCellText;
- strCellText.TrimLeft(); strCellText.TrimRight();
- SendMessageToParent(TargetCell.row, TargetCell.col, GVN_BEGINLABELEDIT);
- SetItemText(TargetCell.row, TargetCell.col, strCellText);
- SetModified(TRUE, TargetCell.row, TargetCell.col);
- SendMessageToParent(TargetCell.row, TargetCell.col, GVN_ENDLABELEDIT);
- // Make sure cell is not selected to avoid data loss
- SetItemState(TargetCell.row, TargetCell.col,
- GetItemState(TargetCell.row, TargetCell.col) & ~GVIS_SELECTED);
- }
- pszCellText = _tcstok(NULL, _T("\t,\n"));
- nColumn++;
- }
- strLine.ReleaseBuffer();
- strLine = strNext;
- nLine++;
- } while (nIndex >= 0);
- strText.UnlockBuffer();
- Invalidate();
- return TRUE;
- }
- #endif
- #ifndef GRIDCONTROL_NO_DRAGDROP
- void CGridCtrl::OnBeginDrag()
- {
- if (!m_bAllowDragAndDrop) return;
- COleDataSource* pSource = CopyTextFromGrid();
- if (pSource)
- {
- SendMessageToParent(GetSelectedCellRange().GetTopLeft().row,
- GetSelectedCellRange().GetTopLeft().col,
- GVN_BEGINDRAG);
- m_MouseMode = MOUSE_DRAGGING;
- DROPEFFECT dropEffect = pSource->DoDragDrop(DROPEFFECT_COPY|DROPEFFECT_MOVE);
- if (dropEffect & DROPEFFECT_MOVE)
- CutSelectedText();
- if (pSource)
- delete pSource; // Did not pass source to clipboard, so must delete
- }
- }
- DROPEFFECT CGridCtrl::OnDragOver(COleDataObject* pDataObject, DWORD dwKeyState,
- CPoint point)
- {
- // Any text data available for us?
- if (!m_bAllowDragAndDrop || !IsEditable() || !pDataObject->IsDataAvailable(CF_TEXT))
- return DROPEFFECT_NONE;
- // Find which cell we are over and drop-highlight it
- CCellID cell = GetCellFromPt(point, FALSE);
- // If not valid, set the previously drop-highlighted cell as no longer drop-highlighted
- if (!IsValid(cell))
- {
- OnDragLeave();
- m_LastDragOverCell = CCellID(-1,-1);
- return DROPEFFECT_NONE;
- }
- if (!IsCellEditable(cell))
- return DROPEFFECT_NONE;
- // Have we moved over a different cell than last time?
- if (cell != m_LastDragOverCell)
- {
- // Set the previously drop-highlighted cell as no longer drop-highlighted
- if (IsValid(m_LastDragOverCell)) {
- UINT nState = GetItemState(m_LastDragOverCell.row, m_LastDragOverCell.col);
- SetItemState(m_LastDragOverCell.row, m_LastDragOverCell.col,
- nState & ~GVIS_DROPHILITED);
- RedrawCell(m_LastDragOverCell);
- }
- m_LastDragOverCell = cell;
- // Set the new cell as drop-highlighted
- if (IsValid(m_LastDragOverCell)) {
- UINT nState = GetItemState(m_LastDragOverCell.row, m_LastDragOverCell.col);
- SetItemState(m_LastDragOverCell.row, m_LastDragOverCell.col,
- nState | GVIS_DROPHILITED);
- RedrawCell(m_LastDragOverCell);
- }
- }
- // Return an appropraite value of DROPEFFECT so mouse cursor is set properly
- if (dwKeyState & MK_CONTROL)
- return DROPEFFECT_COPY;
- else
- return DROPEFFECT_MOVE;
- }
- DROPEFFECT CGridCtrl::OnDragEnter(COleDataObject* pDataObject, DWORD dwKeyState,
- CPoint point)
- {
- // Any text data available for us?
- if (!m_bAllowDragAndDrop || !pDataObject->IsDataAvailable(CF_TEXT))
- return DROPEFFECT_NONE;
- // Find which cell we are over and drop-highlight it
- m_LastDragOverCell = GetCellFromPt(point, FALSE);
- if (!IsValid(m_LastDragOverCell))
- return DROPEFFECT_NONE;
- if (!IsCellEditable(m_LastDragOverCell))
- return DROPEFFECT_NONE;
- if (IsValid(m_LastDragOverCell))
- {
- UINT nState = GetItemState(m_LastDragOverCell.row, m_LastDragOverCell.col);
- SetItemState(m_LastDragOverCell.row, m_LastDragOverCell.col,
- nState | GVIS_DROPHILITED);
- RedrawCell(m_LastDragOverCell);
- }
- // Return an appropraite value of DROPEFFECT so mouse cursor is set properly
- if (dwKeyState & MK_CONTROL)
- return DROPEFFECT_COPY;
- else
- return DROPEFFECT_MOVE;
- }
- void CGridCtrl::OnDragLeave()
- {
- // Set the previously drop-highlighted cell as no longer drop-highlighted
- if (IsValid(m_LastDragOverCell)) {
- UINT nState = GetItemState(m_LastDragOverCell.row, m_LastDragOverCell.col);
- SetItemState(m_LastDragOverCell.row, m_LastDragOverCell.col,
- nState & ~GVIS_DROPHILITED);
- RedrawCell(m_LastDragOverCell);
- }
- }
- BOOL CGridCtrl::OnDrop(COleDataObject* pDataObject, DROPEFFECT /*dropEffect*/,
- CPoint /* point */)
- {
- BOOL bResult = FALSE;
- if (!m_bAllowDragAndDrop || !IsCellEditable(m_LastDragOverCell))
- return bResult;
- m_MouseMode = MOUSE_NOTHING;
- OnDragLeave();
- return PasteTextToGrid(m_LastDragOverCell, pDataObject);
- }
- #endif
- #ifndef GRIDCONTROL_NO_CLIPBOARD
- void CGridCtrl::OnEditCut()
- {
- if (!IsEditable())
- return;
- COleDataSource* pSource = CopyTextFromGrid();
- if (!pSource) return;
- pSource->SetClipboard();
- CutSelectedText();
- }
- void CGridCtrl::OnEditCopy()
- {
- COleDataSource* pSource = CopyTextFromGrid();
- if (!pSource) return;
- pSource->SetClipboard();
- }
- void CGridCtrl::OnEditPaste()
- {
- if (!IsEditable())
- return;
- // Get the Focus cell, or if none, get the topleft (non-fixed) cell
- CCellID cell = GetFocusCell();
- if (!IsValid(cell)) cell = GetTopleftNonFixedCell();
- if (!IsValid(cell)) return;
- // Attach a COleDataObject to the clipboard and paste the data to the grid
- COleDataObject obj;
- if (obj.AttachClipboard())
- PasteTextToGrid(cell, &obj);
- }
- #endif
- void CGridCtrl::OnEditSelectAll()
- {
- SelectAllCells();
- }
- #ifndef GRIDCONTROL_NO_CLIPBOARD
- void CGridCtrl::OnUpdateEditCopy(CCmdUI* pCmdUI)
- {
- CCellRange Selection = GetSelectedCellRange();
- pCmdUI->Enable(Selection.Count() && IsValid(Selection));
- }
- void CGridCtrl::OnUpdateEditCut(CCmdUI* pCmdUI)
- {
- CCellRange Selection = GetSelectedCellRange();
- pCmdUI->Enable(IsEditable() && Selection.Count() && IsValid(Selection));
- }
- void CGridCtrl::OnUpdateEditPaste(CCmdUI* pCmdUI)
- {
- CCellID cell = GetFocusCell();
- BOOL bCanPaste = IsValid(cell) && IsCellEditable(cell) &&
- ::IsClipboardFormatAvailable(CF_TEXT);
- pCmdUI->Enable(bCanPaste);
- }
- #endif
- void CGridCtrl::OnUpdateEditSelectAll(CCmdUI* pCmdUI)
- {
- pCmdUI->Enable(m_bEnableSelection);
- }
- ////////////////////////////////////////////////////////////////////////////////////////
- // hittest-like functions
- BOOL CGridCtrl::MouseOverRowResizeArea(CPoint& point) const
- {
- if (point.x >= GetFixedColumnWidth())
- return FALSE;
-
- CCellID idCurrentCell = GetCellFromPt(point);
- CPoint start;
- if (!GetCellOrigin(idCurrentCell, &start))
- return FALSE;
-
- int endy = start.y + GetRowHeight(idCurrentCell.row);
-
- if ((point.y - start.y <= m_nResizeCaptureRange && idCurrentCell.row != 0) ||
- endy - point.y <= m_nResizeCaptureRange)
- {
- return TRUE;
- }
- else
- return FALSE;
- }
- BOOL CGridCtrl::MouseOverColumnResizeArea(CPoint& point) const
- {
- if (point.y >= GetFixedRowHeight())
- return FALSE;
-
- CCellID idCurrentCell = GetCellFromPt(point);
- CPoint start;
- if (!GetCellOrigin(idCurrentCell, &start))
- return FALSE;
-
- int endx = start.x + GetColumnWidth(idCurrentCell.col);
-
- if ((point.x - start.x <= m_nResizeCaptureRange && idCurrentCell.col != 0) ||
- endx - point.x <= m_nResizeCaptureRange)
- {
- return TRUE;
- }
- else
- return FALSE;
- }
- // Get cell from point
- CCellID CGridCtrl::GetCellFromPt(CPoint point, BOOL bAllowFixedCellCheck /*=TRUE*/) const
- {
- CCellID idTopLeft = GetTopleftNonFixedCell();
- CCellID cellID; // return value
- // calculate column index
- int fixedColWidth = GetFixedColumnWidth();
- if (point.x < 0 || (!bAllowFixedCellCheck && point.x < fixedColWidth)) // not in window
- cellID.col = -1;
- else if (point.x < fixedColWidth) // in fixed col
- {
- int xpos = 0;
- int col = 0;
- for (col = 0; col < m_nFixedCols; col++)
- {
- xpos += GetColumnWidth(col);
- if (xpos > point.x) break;
- }
- cellID.col = col;
- }
- else // in non-fixed col
- {
- int xpos = fixedColWidth;
- int col = 0;
- for (col = idTopLeft.col; col < GetColumnCount(); col++)
- {
- xpos += GetColumnWidth(col);
- if (xpos > point.x) break;
- }
- if (col >= GetColumnCount())
- cellID.col = -1;
- else
- cellID.col = col;
- }
-
- // calculate row index
- int fixedRowHeight = GetFixedRowHeight();
- if (point.y < 0 || (!bAllowFixedCellCheck && point.y < fixedRowHeight)) // not in window
- cellID.row = -1;
- else if (point.y < fixedRowHeight) // in fixed col
- {
- int ypos = 0;
- int row = 0;
- for (row = 0; row < m_nFixedRows; row++)
- {
- ypos += GetRowHeight(row);
- if (ypos > point.y) break;
- }
- cellID.row = row;
- }
- else
- {
- int ypos = fixedRowHeight;
- int row = 0;
- for (row = idTopLeft.row; row < GetRowCount(); row++)
- {
- ypos += GetRowHeight(row);
- if (ypos > point.y) break;
- }
- if (row >= GetRowCount())
- cellID.row = -1;
- else
- cellID.row = row;
- }
- return cellID;
- }
- ////////////////////////////////////////////////////////////////////////////////
- // CGridCtrl cellrange functions
- CCellID CGridCtrl::GetTopleftNonFixedCell() const
- {
- int nVertScroll = GetScrollPos(SB_VERT),
- nHorzScroll = GetScrollPos(SB_HORZ);
- int nColumn = m_nFixedCols, nRight = 0;
- while (nRight < nHorzScroll && nColumn < (GetColumnCount()-1))
- nRight += GetColumnWidth(nColumn++);
- int nRow = m_nFixedRows, nTop = 0;
- while (nTop < nVertScroll && nRow < (GetRowCount()-1))
- nTop += GetRowHeight(nRow++);
- //TRACE("TopLeft cell is row %d, col %d\n",nRow, nColumn);
- return CCellID(nRow, nColumn);
- }
- // This gets even partially visible cells
- CCellRange CGridCtrl::GetVisibleNonFixedCellRange(LPRECT pRect /*=NULL*/) const
- {
- CRect rect;
- GetClientRect(rect);
- CCellID idTopLeft = GetTopleftNonFixedCell();
- // calc bottom
- int bottom = GetFixedRowHeight();
- int i = 0;
- for (i = idTopLeft.row; i < GetRowCount(); i++)
- {
- bottom += GetRowHeight(i);
- if (bottom >= rect.bottom) {
- bottom = rect.bottom;
- break;
- }
- }
- int maxVisibleRow = min(i, GetRowCount() - 1);
-
- // calc right
- int right = GetFixedColumnWidth();
- for (i = idTopLeft.col; i < GetColumnCount(); i++)
- {
- right += GetColumnWidth(i);
- if (right >= rect.right) {
- right = rect.right;
- break;
- }
- }
- int maxVisibleCol = min(i, GetColumnCount() - 1);
- if (pRect) {
- pRect->left = pRect->top = 0;
- pRect->right = right;
- pRect->bottom = bottom;
- }
- return CCellRange(idTopLeft.row, idTopLeft.col, maxVisibleRow, maxVisibleCol);
- }
- // used by ResetScrollBars() - This gets only fully visible cells
- CCellRange CGridCtrl::GetUnobstructedNonFixedCellRange() const
- {
- CRect rect;
- GetClientRect(rect);
- CCellID idTopLeft = GetTopleftNonFixedCell();
- // calc bottom
- int bottom = GetFixedRowHeight();
- int i = 0;
- for (i = idTopLeft.row; i < GetRowCount(); i++)
- {
- bottom += GetRowHeight(i);
- if (bottom >= rect.bottom) break;
- }
- int maxVisibleRow = min(i, GetRowCount() - 1);
- if (maxVisibleRow > 0 && bottom > rect.bottom) maxVisibleRow--;
-
- // calc right
- int right = GetFixedColumnWidth();
- for (i = idTopLeft.col; i < GetColumnCount(); i++)
- {
- right += GetColumnWidth(i);
- if (right >= rect.right) break;
- }
- int maxVisibleCol = min(i, GetColumnCount() - 1);
- if (maxVisibleCol > 0 && right > rect.right) maxVisibleCol--;
- return CCellRange(idTopLeft.row, idTopLeft.col, maxVisibleRow, maxVisibleCol);
- }
- // Returns the minimum bounding range of the current selection
- // If no selection, then the returned CCellRange will be invalid
- CCellRange CGridCtrl::GetSelectedCellRange() const
- {
- CCellRange Selection(GetRowCount(), GetColumnCount(), -1,-1);
- for (POSITION pos = m_SelectedCellMap.GetStartPosition(); pos != NULL; )
- {
- DWORD key;
- CCellID cell;
- m_SelectedCellMap.GetNextAssoc(pos, key, (CCellID&)cell);
- Selection.SetMinRow( min(Selection.GetMinRow(), cell.row) );
- Selection.SetMinCol( min(Selection.GetMinCol(), cell.col) );
- Selection.SetMaxRow( max(Selection.GetMaxRow(), cell.row) );
- Selection.SetMaxCol( max(Selection.GetMaxCol(), cell.col) );
- }
- return Selection;
- }
- // Returns ALL the cells in the grid
- CCellRange CGridCtrl::GetCellRange() const
- {
- return CCellRange(0, 0, GetRowCount() - 1, GetColumnCount() - 1);
- }
- void CGridCtrl::ResetSelectedRange()
- {
- SetSelectedRange(-1,-1,-1,-1);
- SetFocusCell(-1,-1);
- }
- // Get/Set scroll position using 32 bit functions
- int CGridCtrl::GetScrollPos32(int nBar, BOOL bGetTrackPos /* = FALSE */)
- {
- SCROLLINFO si;
- si.cbSize = sizeof(SCROLLINFO);
- if (bGetTrackPos)
- {
- if (GetScrollInfo(nBar, &si, SIF_TRACKPOS))
- return si.nTrackPos;
- }
- else
- {
- if (GetScrollInfo(nBar, &si, SIF_POS))
- return si.nPos;
- }
- return 0;
- }
- BOOL CGridCtrl::SetScrollPos32(int nBar, int nPos, BOOL bRedraw /* = TRUE */)
- {
- SCROLLINFO si;
- si.cbSize = sizeof(SCROLLINFO);
- si.fMask = SIF_POS;
- si.nPos = nPos;
- return SetScrollInfo(nBar, &si, bRedraw);
- }
- void CGridCtrl::ResetScrollBars()
- {
- if (!m_bAllowDraw || !::IsWindow(GetSafeHwnd()))
- return;
- CRect rect;
- GetClientRect(rect);
- rect.left += GetFixedColumnWidth();
- rect.top += GetFixedRowHeight();
- if (rect.left >= rect.right || rect.top >= rect.bottom) return;
- CRect VisibleRect(GetFixedColumnWidth(), GetFixedRowHeight(), rect.right, rect.bottom);
- CRect VirtualRect(GetFixedColumnWidth(), GetFixedRowHeight(), GetVirtualWidth(), GetVirtualHeight());
- CCellRange visibleCells = GetUnobstructedNonFixedCellRange();
- if (!IsValid(visibleCells)) return;
- SCROLLINFO si;
- si.cbSize = sizeof(SCROLLINFO);
- si.fMask = SIF_PAGE;
- si.nPage = VisibleRect.Width(); SetScrollInfo(SB_HORZ, &si, FALSE);
- si.nPage = VisibleRect.Height(); SetScrollInfo(SB_VERT, &si, FALSE);
- if (VisibleRect.Height() < VirtualRect.Height())
- m_nVScrollMax = VirtualRect.Height()-1; //+ GetRowHeight(visibleCells.GetTopLeft().row);
- else
- m_nVScrollMax = 0;
- if (VisibleRect.Width() < VirtualRect.Width())
- m_nHScrollMax = VirtualRect.Width()-1; //+ GetColumnWidth(visibleCells.GetTopLeft().col);
- else
- m_nHScrollMax = 0;
- ASSERT(m_nVScrollMax < INT_MAX && m_nHScrollMax < INT_MAX); // This should be fine :)
- SetScrollRange(SB_VERT, 0, m_nVScrollMax, TRUE);
- SetScrollRange(SB_HORZ, 0, m_nHScrollMax, TRUE);
- }
- ////////////////////////////////////////////////////////////////////////////////////
- // Row/Column position functions
- // returns the top left point of the cell. Returns FALSE if cell not visible.
- BOOL CGridCtrl::GetCellOrigin(int nRow, int nCol, LPPOINT p) const
- {
- int i;
- if (!IsValid(nRow, nCol)) return FALSE;
- CCellID idTopLeft;
- if (nCol >= m_nFixedCols || nRow >= m_nFixedRows)
- idTopLeft = GetTopleftNonFixedCell();
- if ((nRow >= m_nFixedRows && nRow < idTopLeft.row) ||
- (nCol>= m_nFixedCols && nCol < idTopLeft.col))
- return FALSE;
- p->x = 0;
- if (nCol < m_nFixedCols) // is a fixed column
- for (i = 0; i < nCol; i++)
- p->x += GetColumnWidth(i);
- else { // is a scrollable data column
- for (i = 0; i < m_nFixedCols; i++)
- p->x += GetColumnWidth(i);
- for (i = idTopLeft.col; i < nCol; i++)
- p->x += GetColumnWidth(i);
- }
- p->y = 0;
- if (nRow < m_nFixedRows) // is a fixed row
- for (i = 0; i < nRow; i++)
- p->y += GetRowHeight(i);
- else { // is a scrollable data row
- for (i = 0; i < m_nFixedRows; i++)
- p->y += GetRowHeight(i);
- for (i = idTopLeft.row; i < nRow; i++)
- p->y += GetRowHeight(i);
- }
- return TRUE;
- }
- BOOL CGridCtrl::GetCellOrigin(const CCellID& cell, LPPOINT p) const
- {
- return GetCellOrigin(cell.row, cell.col, p);
- }
- // Returns the bounding box of the cell
- BOOL CGridCtrl::GetCellRect(const CCellID& cell, LPRECT pRect) const
- {
- return GetCellRect(cell.row, cell.col, pRect);
- }
- BOOL CGridCtrl::GetCellRect(int nRow, int nCol, LPRECT pRect) const
- {
- CPoint CellOrigin;
- if (!GetCellOrigin(nRow, nCol, &CellOrigin)) return FALSE;
- pRect->left = CellOrigin.x;
- pRect->top = CellOrigin.y;
- pRect->right = CellOrigin.x + GetColumnWidth(nCol)-1;
- pRect->bottom = CellOrigin.y + GetRowHeight(nRow)-1;
- //TRACE("Row %d, col %d: L %d, T %d, W %d, H %d: %d,%d - %d,%d\n",
- // nRow,nCol, CellOrigin.x, CellOrigin.y, GetColumnWidth(nCol), GetRowHeight(nRow),
- // pRect->left, pRect->top, pRect->right, pRect->bottom);
- return TRUE;
- }
- BOOL CGridCtrl::GetTextRect(const CCellID& cell, LPRECT pRect)
- {
- return GetTextRect(cell.row, cell.col, pRect);
- }
- BOOL CGridCtrl::GetTextRect(int nRow, int nCol, LPRECT pRect)
- {
- GV_ITEM Item;
- Item.mask = GVIF_IMAGE;
- Item.row = nRow;
- Item.col = nCol;
- if (!GetItem(&Item))
- return FALSE;
- if (!GetCellRect(nRow, nCol, pRect))
- return FALSE;
- //pRect->left += m_nMargin;
- //pRect->right -= m_nMargin;
- if (m_pImageList && Item.iImage >= 0) {
- IMAGEINFO Info;
- if (m_pImageList->GetImageInfo(Item.iImage, &Info)) {
- int nImageWidth = Info.rcImage.right-Info.rcImage.left+1;
- pRect->left += nImageWidth+m_nMargin;
- }
- }
- return TRUE;
- }
- // Returns the bounding box of a range of cells
- BOOL CGridCtrl::GetCellRangeRect(const CCellRange& cellRange, LPRECT lpRect) const
- {
- CPoint MinOrigin,MaxOrigin;
- if (!GetCellOrigin(cellRange.GetMinRow(), cellRange.GetMinCol(), &MinOrigin))
- return FALSE;
- if (!GetCellOrigin(cellRange.GetMaxRow(), cellRange.GetMaxCol(), &MaxOrigin))
- return FALSE;
- lpRect->left = MinOrigin.x;
- lpRect->top = MinOrigin.y;
- lpRect->right = MaxOrigin.x + GetColumnWidth(cellRange.GetMaxCol()-1);
- lpRect->bottom = MaxOrigin.y + GetRowHeight(cellRange.GetMaxRow()-1);
- return TRUE;
- }
- ////////////////////////////////////////////////////////////////////////////////////
- // Grid attribute functions
- LRESULT CGridCtrl::OnSetFont(WPARAM hFont, LPARAM /*lParam */)
- {
- LRESULT result = Default();
- // Get the logical font
- LOGFONT lf;
- if (!GetObject((HFONT) hFont, sizeof(LOGFONT), &lf))
- return result;
-
- // Store font as the global default
- memcpy(&m_Logfont, &lf, sizeof(LOGFONT));
-
- // reset all cells' fonts
- for (int row = 0; row < GetRowCount(); row++)
- for (int col = 0; col < GetColumnCount(); col++)
- SetItemFont(row, col, &lf);
- // Get the font size and hence the default cell size
- CDC* pDC = GetDC();
- if (pDC)
- {
- m_Font.DeleteObject();
- m_Font.CreateFontIndirect(&m_Logfont);
- CFont* pOldFont = pDC->SelectObject(&m_Font);
-
- TEXTMETRIC tm;
- pDC->GetTextMetrics(&tm);
-
- m_nMargin = pDC->GetTextExtent(_T(" "),1).cx;
- pDC->SelectObject(pOldFont);
- ReleaseDC(pDC);
- m_nDefCellHeight = tm.tmHeight+tm.tmExternalLeading + 1*m_nMargin;
- m_nDefCellWidth = tm.tmAveCharWidth*12 + 2*m_nMargin;
- }
- if (::IsWindow(GetSafeHwnd()))
- Invalidate();
- return result;
- }
- LRESULT CGridCtrl::OnGetFont(WPARAM /*wParam*/, LPARAM /*lParam*/)
- {
- return (LRESULT) (HFONT) m_Font;
- }
- #ifndef _WIN32_WCE_NO_CURSOR
- BOOL CGridCtrl::OnSetCursor(CWnd* pWnd, UINT nHitTest, UINT message)
- {
- if (nHitTest == HTCLIENT)
- {
- switch (m_MouseMode)
- {
- case MOUSE_OVER_COL_DIVIDE: SetCursor(AfxGetApp()->LoadStandardCursor(IDC_SIZEWE)); break;
- case MOUSE_OVER_ROW_DIVIDE: SetCursor(AfxGetApp()->LoadStandardCursor(IDC_SIZENS)); break;
- #ifndef GRIDCONTROL_NO_DRAGDROP
- case MOUSE_DRAGGING: break;
- #endif
- default: SetCursor(AfxGetApp()->LoadStandardCursor(IDC_ARROW));
- }
- return TRUE;
- }
- return CWnd::OnSetCursor(pWnd, nHitTest, message);
- }
- #endif
- ////////////////////////////////////////////////////////////////////////////////////
- // Row/Column count functions
- BOOL CGridCtrl::SetFixedRowCount(int nFixedRows)
- {
- ASSERT(nFixedRows >= 0);
- if (nFixedRows > GetRowCount())
- if (!SetRowCount(nFixedRows)) return FALSE;
- if (m_idCurrentCell.row < nFixedRows)
- SetFocusCell(-1,-1);
- m_nFixedRows = nFixedRows;
- if (GetSafeHwnd() && m_bAllowDraw)
- Invalidate();
- return TRUE;
- }
- BOOL CGridCtrl::SetFixedColumnCount(int nFixedCols)
- {
- ASSERT(nFixedCols >= 0);
- if (nFixedCols > GetColumnCount())
- if (!SetColumnCount(nFixedCols)) return FALSE;
- if (m_idCurrentCell.col < nFixedCols)
- SetFocusCell(-1,-1);
- m_nFixedCols = nFixedCols;
- if (GetSafeHwnd() && m_bAllowDraw)
- Invalidate();
- return TRUE;
- }
- BOOL CGridCtrl::SetRowCount(int nRows)
- {
- ASSERT(nRows >= 0);
- if (nRows == GetRowCount()) return TRUE;
- if (nRows < m_nFixedRows)
- m_nFixedRows = nRows;
- if (m_idCurrentCell.row >= nRows)
- SetFocusCell(-1,-1);
- int addedRows = nRows - GetRowCount();
- // If we are about to lose rows, then we need to delete the GridCell objects
- // in each column within each row
- if (addedRows < 0) {
- for (int row = nRows; row < m_nRows; row++)
- {
- // Delete cells
- for (int col = 0; col < m_nCols; col++)
- {
- CGridCell* pCell = GetCell(row, col);
- if (pCell) {
- EmptyCell(pCell, row, col);
- delete pCell;
- }
- }
- // Delete rows
- GRID_ROW* pRow = m_RowData[row];
- if (pRow) delete pRow;
- }
- }
- // Change the number of rows.
- m_nRows = nRows;
- m_RowData.SetSize(m_nRows);
- m_arRowHeights.SetSize(nRows);
- // If we have just added rows, we need to construct new elements for each cell
- // and set the default row height
- if (addedRows > 0) {
- // initialize row heights and data
- int startRow = nRows - addedRows;
- for (int row = startRow; row < GetRowCount(); row++) {
- m_arRowHeights[row] = m_nDefCellHeight;
- m_RowData[row] = new GRID_ROW;
- m_RowData[row]->SetSize(m_nCols);
- for (int col = 0; col < m_nCols; col++)
- {
- GRID_ROW* pRow = m_RowData[row];
- if (pRow) pRow->SetAt(col, CreateCell(row, col));
- }
- }
- }
- //else
- // ResetSelectedRange();
-
- if (GetSafeHwnd() && m_bAllowDraw)
- {
- ResetScrollBars();
- Invalidate();
- }
- return TRUE;
- }
- BOOL CGridCtrl::SetColumnCount(int nCols)
- {
- ASSERT(nCols >= 0);
- if (nCols == GetColumnCount()) return TRUE;
- if (nCols < m_nFixedCols)
- m_nFixedCols = nCols;
- if (m_idCurrentCell.col >= nCols)
- SetFocusCell(-1,-1);
- int addedCols = nCols - GetColumnCount();
- // If we are about to lose columns, then we need to delete the GridCell objects
- // within each column
- if (addedCols < 0) {
- for (int row = 0; row < m_nRows; row++)
- for (int col = nCols; col < GetColumnCount(); col++)
- {
- CGridCell* pCell = GetCell(row, col);
- if (pCell) {
- EmptyCell(pCell, row, col);
- delete pCell;
- }
- }
- }
- // Change the number of columns.
- m_nCols = nCols;
- m_arColWidths.SetSize(nCols);
- // Change the number of columns in each row.
- for (int i = 0; i < m_nRows; i++)
- if (m_RowData[i]) m_RowData[i]->SetSize(nCols);
- // If we have just added columns, we need to construct new elements for each cell
- // and set the default column width
- if (addedCols > 0)
- {
- // initialized column widths
- int startCol = nCols - addedCols;
- int col = 0;
- for (col = startCol; col < GetColumnCount(); col++)
- m_arColWidths[col] = m_nDefCellWidth;
- // initialise column data
- for (int row = 0; row < m_nRows; row++)
- for (col = startCol; col < GetColumnCount(); col++)
- {
- GRID_ROW* pRow = m_RowData[row];
- if (pRow) pRow->SetAt(col, CreateCell(row,col));
- }
- }
- //else // check for selected cell ranges
- // ResetSelectedRange();
-
- if (GetSafeHwnd() && m_bAllowDraw)
- {
- ResetScrollBars();
- Invalidate();
- }
- return TRUE;
- }
- // Insert a column at a given position, or add to end of columns (if nColumn = -1)
- int CGridCtrl::InsertColumn(LPCTSTR strHeading,
- UINT nFormat /* = DT_CENTER|DT_VCENTER|DT_SINGLELINE */,
- int nColumn /* = -1 */)
- {
- // If the insertion is for a specific column, check it's within range.
- if (nColumn >= 0 && nColumn >= GetColumnCount())
- return -1;
- ResetSelectedRange();
- // Gotta be able to at least _see_ some of the column.
- if (m_nRows < 1)
- SetRowCount(1);
- if (nColumn < 0)
- {
- nColumn = m_nCols;
- m_arColWidths.Add(0);
- for (int row = 0; row < m_nRows; row++)
- {
- GRID_ROW* pRow = m_RowData[row];
- if (!pRow) return -1;
- pRow->Add(CreateCell(row, nColumn));
- }
- }
- else
- {
- m_arColWidths.InsertAt(nColumn, (UINT)0);
- for (int row = 0; row < m_nRows; row++)
- {
- GRID_ROW* pRow = m_RowData[row];
- if (!pRow) return -1;
- pRow->InsertAt(nColumn, CreateCell(row, nColumn));
- }
- }
- m_nCols++;
- // Initialise column data
- SetItemText(0, nColumn, strHeading);
- for (int row = 0; row < m_nRows; row++)
- {
- SetItemFormat(row, nColumn, nFormat);
- }
- // initialized column width
- m_arColWidths[nColumn] = GetTextExtent(strHeading).cx;
- if (m_idCurrentCell.col != -1 && nColumn < m_idCurrentCell.col)
- m_idCurrentCell.col++;
-
- ResetScrollBars();
- return nColumn;
- }
- // Insert a row at a given position, or add to end of rows (if nRow = -1)
- int CGridCtrl::InsertRow(LPCTSTR strHeading, int nRow /* = -1 */)
- {
- // If the insertion is for a specific row, check it's within range.
- if (nRow >= 0 && nRow > GetRowCount())
- return -1;
- ResetSelectedRange();
- // Gotta be able to at least _see_ some of the row.
- if (m_nCols < 1)
- SetColumnCount(1);
- // Adding a row to the bottom
- if (nRow < 0)
- {
- nRow = m_nRows;
- m_arRowHeights.Add(0);
- m_RowData.Add(new GRID_ROW);
- }
- else
- {
- m_arRowHeights.InsertAt(nRow, (UINT)0);
- m_RowData.InsertAt(nRow, new GRID_ROW);
- }
-
- m_nRows++;
- m_RowData[nRow]->SetSize(m_nCols);
- // Initialise cell data
- for (int col = 0; col < m_nCols; col++)
- {
- GRID_ROW* pRow = m_RowData[nRow];
- if (!pRow) return -1;
- pRow->SetAt(col, CreateCell(nRow, col));
- }
- // Set row title
- SetItemText(nRow, 0, strHeading);
- // initialized row height
- m_arRowHeights[nRow] = GetTextExtent(strHeading).cy;
- if (m_idCurrentCell.row != -1 && nRow < m_idCurrentCell.row)
- m_idCurrentCell.row++;
-
- ResetScrollBars();
- return nRow;
- }
- // Creates a new grid cell and performs any necessary initialisation
- CGridCell* CGridCtrl::CreateCell(int nRow, int nCol)
- {
- CGridCell* pCell = new CGridCell;
- if (!pCell)
- return NULL;
-
- // Make format same as cell above
- if (nRow > 0 && nCol >= 0 && nCol < m_nCols)
- pCell->nFormat = GetItemFormat(nRow-1, nCol);
-
- // Make font default grid font
- memcpy(&(pCell->lfFont), &m_Logfont, sizeof(LOGFONT));
-
- return pCell;
- }
- // Performs any cell cleanup necessary to maintain grid integrity
- void CGridCtrl::EmptyCell(CGridCell* pCell, int nRow, int nCol)
- {
- // Set the cells state to 0. If the cell is selected, this
- // will remove the cell from the selected list.
- SetItemState(nRow, nCol, 0);
- // Empty strings
- pCell->szText.Empty();
- }
- BOOL CGridCtrl::DeleteColumn(int nColumn)
- {
- if (nColumn < 0 || nColumn >= GetColumnCount())
- return FALSE;
- ResetSelectedRange();
- for (int row = 0; row < GetRowCount(); row++)
- {
- GRID_ROW* pRow = m_RowData[row];
- if (!pRow) return FALSE;
- CGridCell* pCell = pRow->GetAt(nColumn);
- if (pCell) {
- EmptyCell(pCell, row, nColumn);
- delete pCell;
- }
- pRow->RemoveAt(nColumn);
- }
- m_arColWidths.RemoveAt(nColumn);
- m_nCols--;
- if (nColumn < m_nFixedCols) m_nFixedCols--;
- if (nColumn == m_idCurrentCell.col)
- m_idCurrentCell.row = m_idCurrentCell.col = -1;
- else if (nColumn < m_idCurrentCell.col)
- m_idCurrentCell.col--;
- ResetScrollBars();
- return TRUE;
- }
- BOOL CGridCtrl::DeleteRow(int nRow)
- {
- if ( nRow < 0 || nRow >= GetRowCount())
- return FALSE;
- GRID_ROW* pRow = m_RowData[nRow];
- if (!pRow) return FALSE;
- ResetSelectedRange();
- for (int col = 0; col < GetColumnCount(); col++)
- {
- CGridCell* pCell = pRow->GetAt(col);
- if (pCell) {
- EmptyCell(pCell, nRow, col);
- delete pCell;
- }
- }
- delete pRow;
- m_RowData.RemoveAt(nRow);
- m_arRowHeights.RemoveAt(nRow);
- m_nRows--;
- if (nRow < m_nFixedRows) m_nFixedRows--;
- if (nRow == m_idCurrentCell.row)
- m_idCurrentCell.row = m_idCurrentCell.col = -1;
- else if (nRow < m_idCurrentCell.row)
- m_idCurrentCell.row--;
- ResetScrollBars();
- return TRUE;
- }
- // Handy function that removes all non-fixed rows
- BOOL CGridCtrl::DeleteNonFixedRows()
- {
- int nFixed = GetFixedRowCount();
- int nCount = GetRowCount();
- // Delete all data rows
- for (int nRow = nCount; nRow >= nFixed; nRow--)
- DeleteRow(nRow);
- return TRUE;
- }
- // Removes all rows, columns and data from the grid.
- BOOL CGridCtrl::DeleteAllItems()
- {
- ResetSelectedRange();
- m_arColWidths.RemoveAll();
- m_arRowHeights.RemoveAll();
- // Delete all cells in the grid
- for (int row = 0; row < m_nRows; row++)
- {
- GRID_ROW* pRow = m_RowData[row];
- if (!pRow) continue;
- for (int col = 0; col < m_nCols; col++)
- {
- CGridCell* pCell = pRow->GetAt(col);
- if (pCell) {
- EmptyCell(pCell, row, col); // TODO - this is a bit of a performance hit.
- delete pCell; // better to call m_SelectedCells.RemoveAll()
- } // instead. This is safer for changes though.
- }
- delete pRow;
- }
- // Remove all rows
- m_RowData.RemoveAll();
- m_idCurrentCell.row = m_idCurrentCell.col = -1;
- m_nRows = m_nFixedRows = m_nCols = m_nFixedCols = 0;
- ResetScrollBars();
- return TRUE;
- }
- /////////////////////////////////////////////////////////////////////////////
- // CGridCtrl data functions
- // Set CListCtrl::GetNextItem for details
- CCellID CGridCtrl::GetNextItem(CCellID& cell, int nFlags) const
- {
- if (nFlags & GVNI_ALL)
- { // GVNI_ALL Search whole Grid beginning from cell
- // First row (cell.row) -- ONLY Columns to the right of cell
- // following rows -- ALL Columns
- int row = cell.row , col = cell.col + 1;
- if (row <= 0) row = GetFixedRowCount();
- for (; row < GetRowCount(); row++)
- {
- if (col <= 0 ) col = GetFixedColumnCount();
- for (; col < GetColumnCount(); col++)
- {
- int nState = GetItemState(row, col);
- if (nFlags & GVNI_DROPHILITED && nState & GVIS_DROPHILITED)
- return CCellID(row, col);
- if (nFlags & GVNI_FOCUSED && nState & GVIS_FOCUSED)
- return CCellID(row, col);
- if (nFlags & GVNI_SELECTED && nState & GVIS_SELECTED)
- return CCellID(row, col);
- if (nFlags & GVNI_READONLY && nState & GVIS_READONLY)
- return CCellID(row, col);
- if (nFlags & GVNI_MODIFIED && nState & GVIS_MODIFIED)
- return CCellID(row, col);
- }
- // go to First Column
- col=GetFixedColumnCount();
- }
- }
- else if ((nFlags & GVNI_BELOW) && (nFlags & GVNI_TORIGHT))
- { // GVNI_AREA Search Grid beginning from cell to Lower-Right of Grid
- // Only rows starting with cell.row and below
- // All rows -- ONLY Columns to the right of cell
- int row = cell.row;
- if (row <= 0) row = GetFixedRowCount();
- for (; row < GetRowCount(); row++)
- {
- int col = cell.col + 1;
- if (col <= 0) col = GetFixedColumnCount();
- for (; col < GetColumnCount(); col++)
- {
- int nState = GetItemState(row, col);
- if (nFlags & GVNI_DROPHILITED && nState & GVIS_DROPHILITED)
- return CCellID(row, col);
- if (nFlags & GVNI_FOCUSED && nState & GVIS_FOCUSED)
- return CCellID(row, col);
- if (nFlags & GVNI_SELECTED && nState & GVIS_SELECTED)
- return CCellID(row, col);
- if (nFlags & GVNI_READONLY && nState & GVIS_READONLY)
- return CCellID(row, col);
- if (nFlags & GVNI_MODIFIED && nState & GVIS_MODIFIED)
- return CCellID(row, col);
- }
- }
- }
- else if (nFlags & GVNI_ABOVE)
- {
- for (int row = cell.row-1; row >= GetFixedRowCount(); row--)
- {
- int nState = GetItemState(row, cell.col);
- if (nFlags & GVNI_DROPHILITED && nState & GVIS_DROPHILITED)
- return CCellID(row, cell.col);
- if (nFlags & GVNI_FOCUSED && nState & GVIS_FOCUSED)
- return CCellID(row, cell.col);
- if (nFlags & GVNI_SELECTED && nState & GVIS_SELECTED)
- return CCellID(row, cell.col);
- if (nFlags & GVNI_READONLY && nState & GVIS_READONLY)
- return CCellID(row, cell.col);
- if (nFlags & GVNI_MODIFIED && nState & GVIS_MODIFIED)
- return CCellID(row, cell.col);
- }
- }
- else if (nFlags & GVNI_BELOW)
- {
- for (int row = cell.row+1; row < GetRowCount(); row++)
- {
- int nState = GetItemState(row, cell.col);
- if (nFlags & GVNI_DROPHILITED && nState & GVIS_DROPHILITED)
- return CCellID(row, cell.col);
- if (nFlags & GVNI_FOCUSED && nState & GVIS_FOCUSED)
- return CCellID(row, cell.col);
- if (nFlags & GVNI_SELECTED && nState & GVIS_SELECTED)
- return CCellID(row, cell.col);
- if (nFlags & GVNI_READONLY && nState & GVIS_READONLY)
- return CCellID(row, cell.col);
- if (nFlags & GVNI_MODIFIED && nState & GVIS_MODIFIED)
- return CCellID(row, cell.col);
- }
- }
- else if (nFlags & GVNI_TOLEFT)
- {
- for (int col = cell.col-1; col >= GetFixedColumnCount(); col--)
- {
- int nState = GetItemState(cell.row, col);
- if (nFlags & GVNI_DROPHILITED && nState & GVIS_DROPHILITED)
- return CCellID(cell.row, col);
- if (nFlags & GVNI_FOCUSED && nState & GVIS_FOCUSED)
- return CCellID(cell.row, col);
- if (nFlags & GVNI_SELECTED && nState & GVIS_SELECTED)
- return CCellID(cell.row, col);
- if (nFlags & GVNI_READONLY && nState & GVIS_READONLY)
- return CCellID(cell.row, col);
- if (nFlags & GVNI_MODIFIED && nState & GVIS_MODIFIED)
- return CCellID(cell.row, col);
- }
- }
- else if (nFlags & GVNI_TORIGHT)
- {
- for (int col = cell.col+1; col < GetColumnCount(); col++)
- {
- int nState = GetItemState(cell.row, col);
- if (nFlags & GVNI_DROPHILITED && nState & GVIS_DROPHILITED)
- return CCellID(cell.row, col);
- if (nFlags & GVNI_FOCUSED && nState & GVIS_FOCUSED)
- return CCellID(cell.row, col);
- if (nFlags & GVNI_SELECTED && nState & GVIS_SELECTED)
- return CCellID(cell.row, col);
- if (nFlags & GVNI_READONLY && nState & GVIS_READONLY)
- return CCellID(cell.row, col);
- if (nFlags & GVNI_MODIFIED && nState & GVIS_MODIFIED)
- return CCellID(cell.row, col);
- }
- }
- return CCellID(-1, -1);
- }
- // Sorts on a given column using the cell text
- BOOL CGridCtrl::SortTextItems(int nCol, BOOL bAscending)
- {
- ResetSelectedRange();
- SetFocusCell(-1,-1);
- return SortTextItems(nCol, bAscending, GetFixedRowCount(),-1);
- }
- // recursive sort implementation
- BOOL CGridCtrl::SortTextItems(int nCol, BOOL bAscending, int low, int high)
- {
- if (nCol >= GetColumnCount()) return FALSE;
- if (high == -1) high = GetRowCount() - 1;
- int lo = low;
- int hi = high;
- if( hi <= lo ) return FALSE;
- CString midItem = GetItemText( (lo+hi)/2, nCol );
- // loop through the list until indices cross
- while( lo <= hi )
- {
- // Find the first element that is greater than or equal to the partition
- // element starting from the left Index.
- if( bAscending )
- while (lo < high && GetItemText(lo, nCol) < midItem)
- ++lo;
- else
- while (lo < high && GetItemText(lo, nCol) > midItem)
- ++lo;
- // Find an element that is smaller than or equal to the partition
- // element starting from the right Index.
- if( bAscending )
- while (hi > low && GetItemText(hi, nCol) > midItem)
- --hi;
- else
- while (hi > low && GetItemText(hi, nCol) < midItem)
- --hi;
- // If the indexes have not crossed, swap if the items are not equal
- if (lo <= hi)
- {
- // swap only if the items are not equal
- if (GetItemText(lo, nCol) != GetItemText(hi, nCol))
- {
- for (int col = 0; col < GetColumnCount(); col++)
- {
- CGridCell *pCell = GetCell(lo, col);
- SetCell(lo, col, GetCell(hi, col));
- SetCell(hi, col, pCell);
- }
- UINT nRowHeight = m_arRowHeights[lo];
- m_arRowHeights[lo] = m_arRowHeights[hi];
- m_arRowHeights[hi] = nRowHeight;
- }
- ++lo;
- --hi;
- }
- }
- // If the right index has not reached the left side of array
- // must now sort the left partition.
- if( low < hi )
- SortTextItems(nCol, bAscending, low, hi);
- // If the left index has not reached the right side of array
- // must now sort the right partition.
- if( lo < high )
- SortTextItems(nCol, bAscending, lo, high);
- return TRUE;
- }
- // Sorts on a given column using the supplied compare function (see CListCtrl::SortItems)
- BOOL CGridCtrl::SortItems(PFNLVCOMPARE pfnCompare, int nCol, BOOL bAscending,
- LPARAM data /* = 0 */)
- {
- ResetSelectedRange();
- SetFocusCell(-1,-1);
- return SortItems(pfnCompare, nCol, bAscending, data, GetFixedRowCount(), -1);
- }
- // recursive sort implementation
- BOOL CGridCtrl::SortItems(PFNLVCOMPARE pfnCompare, int nCol, BOOL bAscending, LPARAM data,
- int low, int high)
- {
- if (nCol >= GetColumnCount()) return FALSE;
- if (high == -1) high = GetRowCount() - 1;
- int lo = low;
- int hi = high;
- if( hi <= lo ) return FALSE;
- LPARAM midItem = GetItemData( (lo+hi)/2, nCol );
- // loop through the list until indices cross
- while( lo <= hi )
- {
- // Find the first element that is greater than or equal to the partition
- // element starting from the left Index.
- if( bAscending )
- while (lo < high && pfnCompare(GetItemData(lo, nCol), midItem, data) < 0)
- ++lo;
- else
- while (lo < high && pfnCompare(GetItemData(lo, nCol), midItem, data) > 0)
- ++lo;
- // Find an element that is smaller than or equal to the partition
- // element starting from the right Index.
- if( bAscending )
- while (hi > low && pfnCompare(GetItemData(hi, nCol), midItem, data) > 0)
- --hi;
- else
- while (hi > low && pfnCompare(GetItemData(hi, nCol), midItem, data) < 0)
- --hi;
- // If the indexes have not crossed, swap if the items are not equal
- if (lo <= hi)
- {
- // swap only if the items are not equal
- if (pfnCompare(GetItemData(lo, nCol), GetItemData(hi, nCol), data) != 0)
- {
- for (int col = 0; col < GetColumnCount(); col++)
- {
- CGridCell *pCell = GetCell(lo, col);
- SetCell(lo, col, GetCell(hi, col));
- SetCell(hi, col, pCell);
- }
- UINT nRowHeight = m_arRowHeights[lo];
- m_arRowHeights[lo] = m_arRowHeights[hi];
- m_arRowHeights[hi] = nRowHeight;
- }
- ++lo;
- --hi;
- }
- }
- // If the right index has not reached the left side of array
- // must now sort the left partition.
- if( low < hi )
- SortItems(pfnCompare, nCol, bAscending, data, low, hi);
- // If the left index has not reached the right side of array
- // must now sort the right partition.
- if( lo < high )
- SortItems(pfnCompare, nCol, bAscending, data, lo, high);
- return TRUE;
- }
- /////////////////////////////////////////////////////////////////////////////
- // CGridCtrl data functions
- BOOL CGridCtrl::SetItem(const GV_ITEM* pItem)
- {
- if (!pItem)
- return FALSE;
- CGridCell* pCell = GetCell(pItem->row, pItem->col);
- if (!pCell)
- return FALSE;
- if (pItem->mask & GVIF_TEXT) pCell->szText = pItem->szText;
- if (pItem->mask & GVIF_PARAM) pCell->lParam = pItem->lParam;
- if (pItem->mask & GVIF_IMAGE) pCell->iImage = pItem->iImage;
- if (pItem->mask & GVIF_STATE) pCell->state = pItem->state;
- if (pItem->mask & GVIF_FORMAT) pCell->nFormat = pItem->nFormat;
- if (pItem->mask & GVIF_BKCLR) pCell->crBkClr = pItem->crBkClr;
- if (pItem->mask & GVIF_FGCLR) pCell->crFgClr = pItem->crFgClr;
- if (pItem->mask & GVIF_FONT) memcpy(&(pCell->lfFont), &(pItem->lfFont), sizeof(LOGFONT));
- return TRUE;
- }
- BOOL CGridCtrl::GetItem(GV_ITEM* pItem)
- {
- if (!pItem)
- return FALSE;
- CGridCell* pCell = GetCell(pItem->row, pItem->col);
- if (!pCell)
- return FALSE;
- if (pItem->mask & GVIF_TEXT) pItem->szText = GetItemText(pItem->row, pItem->col);
- if (pItem->mask & GVIF_PARAM) pItem->lParam = pCell->lParam;
- if (pItem->mask & GVIF_IMAGE) pItem->iImage = pCell->iImage;
- if (pItem->mask & GVIF_STATE) pItem->state = pCell->state;
- if (pItem->mask & GVIF_FORMAT) pItem->nFormat = pCell->nFormat;
- if (pItem->mask & GVIF_BKCLR) pItem->crBkClr = pCell->crBkClr;
- if (pItem->mask & GVIF_FGCLR) pItem->crFgClr = pCell->crFgClr;
- if (pItem->mask & GVIF_FONT) memcpy(&(pItem->lfFont), &(pCell->lfFont), sizeof(LOGFONT));
- return TRUE;
- }
- BOOL CGridCtrl::SetItemText(int nRow, int nCol, LPCTSTR str)
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- if (!pCell)
- return FALSE;
- pCell->szText = str;
- return TRUE;
- }
- BOOL CGridCtrl::SetItemData(int nRow, int nCol, LPARAM lParam)
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- if (!pCell)
- return FALSE;
- pCell->lParam = lParam;
- return TRUE;
- }
- LPARAM CGridCtrl::GetItemData(int nRow, int nCol) const
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- if (!pCell)
- return (LPARAM) 0;
- return pCell->lParam;
- }
- BOOL CGridCtrl::SetItemImage(int nRow, int nCol, int iImage)
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- if (!pCell)
- return FALSE;
- pCell->iImage = iImage;
- return TRUE;
- }
- int CGridCtrl::GetItemImage(int nRow, int nCol) const
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- ASSERT(pCell);
- if (!pCell)
- return -1;
- return pCell->iImage;
- }
- BOOL CGridCtrl::SetItemState(int nRow, int nCol, UINT state)
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- ASSERT(pCell);
- if (!pCell)
- return FALSE;
- // If the cell is being unselected, remove it from the selected list
- if ((pCell->state & GVIS_SELECTED) && !(state & GVIS_SELECTED))
- {
- CCellID cell;
- DWORD key = MAKELONG(nRow, nCol);
- if (m_SelectedCellMap.Lookup(key, (CCellID&)cell))
- m_SelectedCellMap.RemoveKey(key);
- }
- // If cell is being selected, add it to the list of selected cells
- else if (!(pCell->state & GVIS_SELECTED) && (state & GVIS_SELECTED))
- {
- CCellID cell(nRow, nCol);
- m_SelectedCellMap.SetAt(MAKELONG(nRow, nCol), cell);
- }
- // Set the cell's state
- pCell->state = state;
- return TRUE;
- }
- UINT CGridCtrl::GetItemState(int nRow, int nCol) const
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- ASSERT(pCell);
- if (!pCell)
- return 0;
- return pCell->state;
- }
- BOOL CGridCtrl::SetItemFormat(int nRow, int nCol, UINT nFormat)
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- ASSERT(pCell);
- if (!pCell)
- return FALSE;
- pCell->nFormat = nFormat;
- return TRUE;
- }
- UINT CGridCtrl::GetItemFormat(int nRow, int nCol) const
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- ASSERT(pCell);
- if (!pCell)
- return 0;
- return pCell->nFormat;
- }
- BOOL CGridCtrl::SetItemBkColour(int nRow, int nCol, COLORREF cr /* = CLR_DEFAULT */)
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- ASSERT(pCell);
- if (!pCell)
- return FALSE;
- pCell->crBkClr = cr;
- return TRUE;
- }
- COLORREF CGridCtrl::GetItemBkColour(int nRow, int nCol) const
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- ASSERT(pCell);
- if (!pCell)
- return 0;
- return pCell->crBkClr;
- }
- BOOL CGridCtrl::SetItemFgColour(int nRow, int nCol, COLORREF cr /* = CLR_DEFAULT */)
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- ASSERT(pCell);
- if (!pCell)
- return FALSE;
- pCell->crFgClr = cr;
- return TRUE;
- }
- COLORREF CGridCtrl::GetItemFgColour(int nRow, int nCol) const
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- ASSERT(pCell);
- if (!pCell)
- return 0;
- return pCell->crFgClr;
- }
- BOOL CGridCtrl::SetItemFont(int nRow, int nCol, LOGFONT* lf)
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- ASSERT(pCell);
- if (!pCell)
- return FALSE;
- memcpy(&(pCell->lfFont), lf, sizeof(LOGFONT));
- return TRUE;
- }
- LOGFONT* CGridCtrl::GetItemFont(int nRow, int nCol)
- {
- CGridCell* pCell = GetCell(nRow, nCol);
- ASSERT(pCell);
- if (!pCell)
- return &m_Logfont;
- return &(pCell->lfFont);
- }
- ////////////////////////////////////////////////////////////////////////////////////
- // Row/Column size functions
- long CGridCtrl::GetVirtualWidth() const
- {
- long lVirtualWidth = 0;
- int iColCount = GetColumnCount();
- for (int i = 0; i < iColCount; i++)
- lVirtualWidth += m_arColWidths[i];
- return lVirtualWidth;
- }
- long CGridCtrl::GetVirtualHeight() const
- {
- long lVirtualHeight = 0;
- int iRowCount = GetRowCount();
- for (int i = 0; i < iRowCount; i++)
- lVirtualHeight += m_arRowHeights[i];
- return lVirtualHeight;
- }
- int CGridCtrl::GetRowHeight(int nRow) const
- {
- ASSERT(nRow >= 0 && nRow < m_nRows);
- if (nRow < 0 || nRow >= m_nRows) return -1;
- return m_arRowHeights[nRow];
- }
- int CGridCtrl::GetColumnWidth(int nCol) const
- {
- ASSERT(nCol >= 0 && nCol < m_nCols);
- if (nCol < 0 || nCol >= m_nCols) return -1;
- return m_arColWidths[nCol];
- }
- BOOL CGridCtrl::SetRowHeight(int nRow, int height)
- {
- ASSERT(nRow >= 0 && nRow < m_nRows && height >= 0);
- if (nRow < 0 || nRow >= m_nRows || height < 0) return FALSE;
- m_arRowHeights[nRow] = height;
- return TRUE;
- }
- BOOL CGridCtrl::SetColumnWidth(int nCol, int width)
- {
- ASSERT(nCol >= 0 && nCol < m_nCols && width >= 0);
- if (nCol < 0 || nCol >= m_nCols || width < 0) return FALSE;
- m_arColWidths[nCol] = width;
- return TRUE;
- }
- int CGridCtrl::GetFixedRowHeight() const
- {
- int nHeight = 0;
- for (int i = 0; i < m_nFixedRows; i++)
- nHeight += GetRowHeight(i);
- return nHeight;
- }
- int CGridCtrl::GetFixedColumnWidth() const
- {
- int nWidth = 0;
- for (int i = 0; i < m_nFixedCols; i++)
- nWidth += GetColumnWidth(i);
- return nWidth;
- }
- BOOL CGridCtrl::AutoSizeColumn(int nCol)
- {
- ASSERT(nCol >= 0 && nCol < m_nCols);
- if (nCol < 0 || nCol >= m_nCols) return FALSE;
- CSize size;
- CDC* pDC = GetDC();
- if (!pDC) return FALSE;
- int nWidth = 0;
- int nNumRows = GetRowCount();
- for (int nRow = 0; nRow < nNumRows; nRow++)
- {
- size = GetCellExtent(nRow, nCol, pDC);
- if (size.cx > nWidth) nWidth = size.cx;
- }
- m_arColWidths[nCol] = nWidth;
- ReleaseDC(pDC);
- ResetScrollBars();
- return TRUE;
- }
- BOOL CGridCtrl::AutoSizeRow(int nRow)
- {
- ASSERT(nRow >= 0 && nRow < m_nRows);
- if (nRow < 0 || nRow >= m_nRows) return FALSE;
- CSize size;
- CDC* pDC = GetDC();
- if (!pDC) return FALSE;
- int nHeight = 0;
- int nNumColumns = GetColumnCount();
- for (int nCol = 0; nCol < nNumColumns; nCol++)
- {
- size = GetCellExtent(nRow, nCol, pDC);
- if (size.cy > nHeight) nHeight = size.cy;
- }
- m_arRowHeights[nRow] = nHeight;
- ReleaseDC(pDC);
- ResetScrollBars();
- return TRUE;
- }
- void CGridCtrl::AutoSizeColumns()
- {
- int nNumColumns = GetColumnCount();
- for (int nCol = 0; nCol < nNumColumns; nCol++)
- AutoSizeColumn(nCol);
- }
- void CGridCtrl::AutoSizeRows()
- {
- int nNumRows = GetRowCount();
- for (int nRow = 0; nRow < nNumRows; nRow++)
- AutoSizeRow(nRow);
- }
- // sizes all rows and columns
- // faster than calling both AutoSizeColumns() and AutoSizeRows()
- void CGridCtrl::AutoSize()
- {
- CDC* pDC = GetDC();
- if (!pDC) return;
- int nNumColumns = GetColumnCount();
- int nNumRows = GetRowCount();
-
- // initialize column widths to zero
- for (int nCol = 0; nCol < nNumColumns; nCol++)
- m_arColWidths[nCol] = 0;
-
- // initialize row heights to zero
- for (int nRow = 0; nRow < nNumRows; nRow++)
- m_arRowHeights[nRow] = 0;
-
- CSize size;
- for (int nCol = 0; nCol < nNumColumns; nCol++)
- for (int nRow = 0; nRow < nNumRows; nRow++)
- {
- size = GetCellExtent(nRow, nCol, pDC);
- if (size.cx > (int) m_arColWidths[nCol]) m_arColWidths[nCol] = size.cx;
- if (size.cy > (int) m_arRowHeights[nRow]) m_arRowHeights[nRow] = size.cy;
- }
-
- ReleaseDC(pDC);
- if (m_bAllowDraw) {
- ResetScrollBars();
- Invalidate();
- }
- }
- void CGridCtrl::ExpandColumnsToFit()
- {
- if (GetColumnCount() <= 0) return;
- CRect rect;
- GetClientRect(rect);
- long virtualWidth = GetVirtualWidth();
- int nDifference = rect.Width() - (int) virtualWidth;
- int nColumnAdjustment = nDifference / GetColumnCount();
- for (int i = 0; i < GetColumnCount(); i++)
- m_arColWidths[i] += nColumnAdjustment;
- if (nDifference > 0)
- {
- int leftOver = nDifference % GetColumnCount();
- for (int i = 0; i < leftOver; i++)
- m_arColWidths[i] += 1;
- }
- else
- {
- int leftOver = (-nDifference) % GetColumnCount();
- for (int i = 0; i < leftOver; i++)
- m_arColWidths[i] -= 1;
- }
- if (m_bAllowDraw)
- Invalidate();
- }
- void CGridCtrl::ExpandRowsToFit()
- {
- if (GetRowCount() <= 0) return;
- CRect rect;
- GetClientRect(rect);
- long virtualHeight = GetVirtualHeight();
- int nDifference = rect.Height() - (int) virtualHeight;
- int nRowAdjustment = nDifference / GetRowCount();
- for (int i = 0; i < GetRowCount(); i++)
- m_arRowHeights[i] += nRowAdjustment;
- if (nDifference > 0)
- {
- int leftOver = nDifference % GetRowCount();
- for (int i = 0; i < leftOver; i++)
- m_arRowHeights[i] += 1;
- }
- else
- {
- int leftOver = (-nDifference) % GetRowCount();
- for (int i = 0; i < leftOver; i++)
- m_arRowHeights[i] -= 1;
- }
- if (m_bAllowDraw)
- Invalidate();
- }
- void CGridCtrl::ExpandToFit()
- {
- ExpandColumnsToFit(); // This will remove any existing horz scrollbar
- ExpandRowsToFit(); // This will remove any existing vert scrollbar
- ExpandColumnsToFit(); // Just in case the first adjustment was with a vert
- // scrollbar in place
- }
- BOOL CGridCtrl::IsCellFixed(int nRow, int nCol)
- {
- return (nRow < GetFixedRowCount() || nCol < GetFixedColumnCount());
- }
- void CGridCtrl::SetModified(BOOL bModified /*=TRUE*/, int nRow /*=-1*/, int nCol /*=-1*/)
- {
- if (nRow > 0 && nCol > 0)
- {
- if (bModified)
- {
- SetItemState(nRow, nCol, GetItemState(nRow, nCol) | GVIS_MODIFIED);
- m_bModified = TRUE;
- }
- else
- SetItemState(nRow, nCol, GetItemState(nRow, nCol) & ~GVIS_MODIFIED);
- }
- else
- m_bModified = bModified;
- if (!m_bModified)
- {
- for (int row = 0; row < GetRowCount(); row++)
- for (int col = 0; col < GetColumnCount(); col++)
- SetItemState(row, col, GetItemState(row, col) & ~GVIS_MODIFIED);
- }
- }
- BOOL CGridCtrl::GetModified(int nRow /*=-1*/, int nCol /*=-1*/)
- {
- if (nRow > 0 && nCol > 0)
- return ( (GetItemState(nRow, nCol) & GVIS_MODIFIED) == GVIS_MODIFIED );
- else
- return m_bModified;
- }
- /////////////////////////////////////////////////////////////////////////////////////
- // GridCtrl cell visibility tests and invalidation/redraw functions
- // EnsureVisible supplied by Roelf Werkman
- void CGridCtrl::EnsureVisible(int nRow, int nCol)
- {
- CCellRange VisibleCells = GetVisibleNonFixedCellRange();
- int right = nCol - VisibleCells.GetMaxCol();
- int left = VisibleCells.GetMinCol() - nCol;
- int down = nRow - VisibleCells.GetMaxRow();
- int up = VisibleCells.GetMinRow() - nRow;
- while (right > 0)
- {
- SendMessage(WM_HSCROLL, SB_LINERIGHT, 0);
- right--;
- }
- while (left > 0)
- {
- SendMessage(WM_HSCROLL, SB_LINELEFT, 0);
- left--;
- }
- while (down > 0)
- {
- SendMessage(WM_VSCROLL, SB_LINEDOWN, 0);
- down--;
- }
- while (up > 0)
- {
- SendMessage(WM_VSCROLL, SB_LINEUP, 0);
- up--;
- }
- // Move one more if we only see a snall bit of the cell
- CRect rectCell, rectWindow;
- GetCellRect(nRow, nCol, rectCell);
- GetClientRect(rectWindow);
- if (rectCell.right > rectWindow.right)
- SendMessage(WM_HSCROLL, SB_LINERIGHT, 0);
- if (rectCell.bottom > rectWindow.bottom)
- SendMessage(WM_VSCROLL, SB_LINEDOWN, 0);
- }
- BOOL CGridCtrl::IsCellEditable(CCellID &cell) const
- {
- return IsCellEditable(cell.row, cell.col);
- }
- BOOL CGridCtrl::IsCellEditable(int nRow, int nCol) const
- {
- return IsEditable() && ((GetItemState(nRow, nCol) & GVIS_READONLY) != GVIS_READONLY);
- }
- BOOL CGridCtrl::IsCellVisible(CCellID cell) const
- {
- return IsCellVisible(cell.row, cell.col);
- }
- BOOL CGridCtrl::IsCellVisible(int nRow, int nCol) const
- {
- if (!IsWindow(m_hWnd))
- return FALSE;
- int x,y;
- CCellID TopLeft;
- if (nCol >= GetFixedColumnCount() || nRow >= GetFixedRowCount())
- {
- TopLeft = GetTopleftNonFixedCell();
- if (nCol >= GetFixedColumnCount() && nCol < TopLeft.col) return FALSE;
- if (nRow >= GetFixedRowCount() && nRow < TopLeft.row) return FALSE;
- }
- CRect rect;
- GetClientRect(rect);
- if (nCol < GetFixedColumnCount())
- {
- x = 0;
- for (int i = 0; i <= nCol; i++)
- {
- if (x >= rect.right) return FALSE;
- x += GetColumnWidth(i);
- }
- }
- else
- {
- x = GetFixedColumnWidth();
- for (int i = TopLeft.col; i <= nCol; i++)
- {
- if (x >= rect.right) return FALSE;
- x += GetColumnWidth(i);
- }
- }
- if (nRow < GetFixedRowCount())
- {
- y = 0;
- for (int i = 0; i <= nRow; i++)
- {
- if (y >= rect.bottom) return FALSE;
- y += GetRowHeight(i);
- }
- }
- else
- {
- if (nRow < TopLeft.row) return FALSE;
- y = GetFixedRowHeight();
- for (int i = TopLeft.row; i <= nRow; i++)
- {
- if (y >= rect.bottom) return FALSE;
- y += GetRowHeight(i);
- }
- }
- return TRUE;
- }
- BOOL CGridCtrl::InvalidateCellRect(const CCellID& cell)
- {
- if (!::IsWindow(GetSafeHwnd()) || !m_bAllowDraw)
- return FALSE;
- ASSERT(IsValid(cell));
- if (!IsCellVisible(cell.row, cell.col)) return FALSE;
- CRect rect;
- if (!GetCellRect(cell, rect)) return FALSE;
- rect.right++; rect.bottom++;
- InvalidateRect(rect, TRUE);
- return TRUE;
- }
- BOOL CGridCtrl::InvalidateCellRect(const CCellRange& cellRange)
- {
- ASSERT(IsValid(cellRange));
- if (!::IsWindow(GetSafeHwnd()) || !m_bAllowDraw) return FALSE;
- CCellRange visibleCellRange = GetVisibleNonFixedCellRange().Intersect(cellRange);
- CRect rect;
- if (!GetCellRangeRect(visibleCellRange, rect)) return FALSE;
- rect.right++; rect.bottom++;
- InvalidateRect(rect, TRUE);
- return TRUE;
- }
- /////////////////////////////////////////////////////////////////////////////
- // CGridCtrl Mouse stuff
- // Handles mouse wheel notifications
- // Note - if this doesn't work for win95 then use OnRegisteredMouseWheel instead
- #if !defined(_WIN32_WCE) && (_MFC_VER >= 0x0421)
- BOOL CGridCtrl::OnMouseWheel(UINT nFlags, short zDelta, CPoint pt)
- {
- // A m_nRowsPerWheelNotch value less than 0 indicates that the mouse
- // wheel scrolls whole pages, not just lines.
- if (m_nRowsPerWheelNotch == -1)
- {
- int nPagesScrolled = zDelta / 120;
- if (nPagesScrolled > 0)
- for (int i = 0; i < nPagesScrolled; i++)
- PostMessage(WM_VSCROLL, SB_PAGEUP, 0);
- else
- for (int i = 0; i > nPagesScrolled; i--)
- PostMessage(WM_VSCROLL, SB_PAGEDOWN, 0);
- }
- else
- {
- int nRowsScrolled = m_nRowsPerWheelNotch * zDelta / 120;
- if (nRowsScrolled > 0)
- for (int i = 0; i < nRowsScrolled; i++)
- PostMessage(WM_VSCROLL, SB_LINEUP, 0);
- else
- for (int i = 0; i > nRowsScrolled; i--)
- PostMessage(WM_VSCROLL, SB_LINEDOWN, 0);
- }
- return CWnd::OnMouseWheel(nFlags, zDelta, pt);
- }
- #endif // !defined(_WIN32_WCE) && (_MFC_VER >= 0x0421)
- void CGridCtrl::OnMouseMove(UINT nFlags, CPoint point)
- {
- CRect rect;
- GetClientRect(rect);
- #ifndef GRIDCONTROL_NO_DRAGDROP
- // If outside client area, return (unless we are drag n dropping)
- if (m_MouseMode != MOUSE_DRAGGING && !rect.PtInRect(point))
- return;
- #endif
- // If the left mouse button is up, then test to see if row/column sizing is imminent
- if (!(nFlags & MK_LBUTTON)
- || (m_MouseMode == MOUSE_NOTHING && (nFlags & MK_LBUTTON)))
- {
- if (m_bAllowColumnResize && MouseOverColumnResizeArea(point))
- {
- if (m_MouseMode != MOUSE_OVER_COL_DIVIDE)
- {
- SetCursor(AfxGetApp()->LoadStandardCursor(IDC_SIZEWE));
- m_MouseMode = MOUSE_OVER_COL_DIVIDE;
- }
- }
- else if (m_bAllowRowResize && MouseOverRowResizeArea(point))
- {
- if (m_MouseMode != MOUSE_OVER_ROW_DIVIDE)
- {
- SetCursor(AfxGetApp()->LoadStandardCursor(IDC_SIZENS));
- m_MouseMode = MOUSE_OVER_ROW_DIVIDE;
- }
- }
- else if (m_MouseMode != MOUSE_NOTHING)
- {
- SetCursor(AfxGetApp()->LoadStandardCursor(IDC_ARROW));
- m_MouseMode = MOUSE_NOTHING;
- }
- #ifndef GRIDCONTROL_NO_TITLETIPS
- if (m_MouseMode == MOUSE_NOTHING && m_bTitleTips)
- {
- CCellID idCurrentCell = GetCellFromPt(point);
- CRect TextRect, CellRect;
- if (GetTextRect(idCurrentCell.row, idCurrentCell.col, TextRect) &&
- GetCellRect(idCurrentCell.row, idCurrentCell.col, CellRect))
- {
-
-
- // m_TitleTip.Show(TextRect,
- // GetItemText(idCurrentCell.row, idCurrentCell.col),
- // 0,
- // CellRect,
- // GetItemFont(idCurrentCell.row, idCurrentCell.col));
- }
- }
- #endif
- m_LastMousePoint = point;
- return;
- }
- if (!IsValid(m_LeftClickDownCell))
- {
- m_LastMousePoint = point;
- return;
- }
- // If the left mouse button is down, then process appropriately
- if (nFlags & MK_LBUTTON)
- {
- switch(m_MouseMode)
- {
- case MOUSE_SELECT_ALL:
- break;
- case MOUSE_SELECT_COL:
- case MOUSE_SELECT_ROW:
- case MOUSE_SELECT_CELLS:
- {
- CCellID idCurrentCell = GetCellFromPt(point);
- if (!IsValid(idCurrentCell))
- return;
- if (idCurrentCell != GetFocusCell())
- {
- OnSelecting(idCurrentCell);
- //SetFocusCell(max(idCurrentCell.row, m_nFixedRows),
- // max(idCurrentCell.col, m_nFixedCols));
- if (idCurrentCell.row >= m_nFixedRows &&
- idCurrentCell.col >= m_nFixedCols)
- {
- SetFocusCell(idCurrentCell);
- }
- }
- break;
- }
- case MOUSE_SIZING_COL:
- {
- CDC* pDC = GetDC();
- if (!pDC) break;
-
- CRect oldInvertedRect(m_LastMousePoint.x, rect.top,
- m_LastMousePoint.x + 2, rect.bottom);
- pDC->InvertRect(&oldInvertedRect);
- CRect newInvertedRect(point.x, rect.top,
- point.x + 2, rect.bottom);
- pDC->InvertRect(&newInvertedRect);
- ReleaseDC(pDC);
- }
- break;
-
- case MOUSE_SIZING_ROW:
- {
- CDC* pDC = GetDC();
- if (!pDC) break;
-
- CRect oldInvertedRect(rect.left, m_LastMousePoint.y,
- rect.right, m_LastMousePoint.y + 2);
- pDC->InvertRect(&oldInvertedRect);
- CRect newInvertedRect(rect.left, point.y,
- rect.right, point.y + 2);
- pDC->InvertRect(&newInvertedRect);
- ReleaseDC(pDC);
- }
- break;
- #ifndef GRIDCONTROL_NO_DRAGDROP
- case MOUSE_PREPARE_DRAG:
- OnBeginDrag();
- break;
- #endif
- }
- }
- m_LastMousePoint = point;
- }
- void CGridCtrl::OnLButtonDblClk(UINT nFlags, CPoint point)
- {
- if (m_MouseMode == MOUSE_OVER_COL_DIVIDE)
- {
- CCellID cell = GetCellFromPt(point);
- ASSERT(IsValid(cell));
- CPoint start;
- if (!GetCellOrigin(0, cell.col, &start)) return;
- if (point.x - start.x <= m_nResizeCaptureRange) // Clicked right of border
- cell.col--;
- AutoSizeColumn(cell.col);
- Invalidate();
- }
- else if (m_MouseMode == MOUSE_OVER_ROW_DIVIDE)
- {
- CCellID cell = GetCellFromPt(point);
- ASSERT(IsValid(cell));
- CPoint start;
- if (!GetCellOrigin(0, cell.col, &start)) return;
- if (point.y - start.y <= m_nResizeCaptureRange) // Clicked below border
- cell.row--;
- AutoSizeRow(cell.row);
- Invalidate();
- }
- else if (m_MouseMode == MOUSE_NOTHING)
- {
- if (m_LeftClickDownCell.row >= m_nFixedRows &&
- IsValid(m_LeftClickDownCell) &&
- m_LeftClickDownCell.col >= m_nFixedCols)
- {
- OnEditCell(m_idCurrentCell.row, m_idCurrentCell.col, VK_LBUTTON);
- }
- else if (m_bListMode)
- {
- CCellID cell = GetCellFromPt(point);
- if (!IsValid(cell)) return;
- if (cell.row >= m_nFixedRows && cell.col < GetFixedColumnCount())
- OnEditCell(cell.row, cell.col, VK_LBUTTON);
- }
- }
- CWnd::OnLButtonDblClk(nFlags, point);
- }
- void CGridCtrl::OnLButtonDown(UINT nFlags, CPoint point)
- {
- HWND hOldFocusWnd = ::GetFocus();
- m_LeftClickDownPoint = point;
- m_LeftClickDownCell = GetCellFromPt(point);
- if (!IsValid(m_LeftClickDownCell)) return;
- m_SelectionStartCell = (nFlags & MK_SHIFT)? m_idCurrentCell : m_LeftClickDownCell;
- SetFocus(); // Auto-destroy any InPlaceEdit's
- // If the user clicks on the current cell, then prepare to edit it.
- // (If the user moves the mouse, then dragging occurs)
- if (m_LeftClickDownCell == m_idCurrentCell)
- {
- m_MouseMode = MOUSE_PREPARE_EDIT;
- return;
- }
- else if (m_MouseMode != MOUSE_OVER_COL_DIVIDE &&
- m_MouseMode != MOUSE_OVER_ROW_DIVIDE)
- {
- SetFocusCell(-1,-1);
- SetFocusCell(max(m_LeftClickDownCell.row, m_nFixedRows),
- max(m_LeftClickDownCell.col, m_nFixedCols));
- }
- #ifndef GRIDCONTROL_NO_DRAGDROP
- // If the user clicks on a selected cell, then prepare to drag it.
- // (If the user moves the mouse, then dragging occurs)
- if (m_bAllowDragAndDrop && hOldFocusWnd == GetSafeHwnd() &&
- GetItemState(m_LeftClickDownCell.row, m_LeftClickDownCell.col) & GVNI_SELECTED)
- {
- m_MouseMode = MOUSE_PREPARE_DRAG;
- return;
- }
- #endif
- SetCapture();
- if (m_MouseMode == MOUSE_NOTHING)
- {
- if (m_bAllowColumnResize && MouseOverColumnResizeArea(point))
- {
- if (m_MouseMode != MOUSE_OVER_COL_DIVIDE)
- {
- SetCursor(AfxGetApp()->LoadStandardCursor(IDC_SIZEWE));
- m_MouseMode = MOUSE_OVER_COL_DIVIDE;
- }
- }
- else if (m_bAllowRowResize && MouseOverRowResizeArea(point))
- {
- if (m_MouseMode != MOUSE_OVER_ROW_DIVIDE)
- {
- SetCursor(AfxGetApp()->LoadStandardCursor(IDC_SIZENS));
- m_MouseMode = MOUSE_OVER_ROW_DIVIDE;
- }
- }
- //else if (m_MouseMode != MOUSE_NOTHING)
- //{
- // SetCursor(AfxGetApp()->LoadStandardCursor(IDC_ARROW));
- // m_MouseMode = MOUSE_NOTHING;
- //}
- }
- if (m_MouseMode == MOUSE_OVER_COL_DIVIDE) // sizing column
- {
- m_MouseMode = MOUSE_SIZING_COL;
- CPoint start;
- if (!GetCellOrigin(0, m_LeftClickDownCell.col, &start)) return;
- CRect rect;
- GetClientRect(rect);
- CRect invertedRect(point.x, rect.top, point.x + 2, rect.bottom);
- CDC* pDC = GetDC();
- if (pDC) {
- pDC->InvertRect(&invertedRect);
- ReleaseDC(pDC);
- }
- if (point.x - start.x <= m_nResizeCaptureRange) // clicked right of border
- if (!GetCellOrigin(0, --m_LeftClickDownCell.col, &start)) return;
- rect.left = start.x;
- ClientToScreen(rect);
- #ifndef _WIN32_WCE_NO_CURSOR
- ClipCursor(rect);
- #endif
- }
- else if (m_MouseMode == MOUSE_OVER_ROW_DIVIDE) // sizing row
- {
- m_MouseMode = MOUSE_SIZING_ROW;
- CPoint start;
- if (!GetCellOrigin(m_LeftClickDownCell, &start)) return;
- CRect rect;
- GetClientRect(rect);
- CRect invertedRect(rect.left, point.y, rect.right, point.y + 2);
- CDC* pDC = GetDC();
- if (pDC) {
- pDC->InvertRect(&invertedRect);
- ReleaseDC(pDC);
- }
- if (point.y - start.y <= m_nResizeCaptureRange) // clicked below border
- if (!GetCellOrigin(--m_LeftClickDownCell.row, 0, &start)) return;
- rect.top = start.y;
- ClientToScreen(rect);
- #ifndef _WIN32_WCE_NO_CURSOR
- ClipCursor(rect);
- #endif
- }
- else // not sizing or editing -- selecting
- {
- // If Ctrl pressed, save the current cell selection. This will get added
- // to the new cell selection at the end of the cell selection process
- m_PrevSelectedCellMap.RemoveAll();
- if (nFlags & MK_CONTROL) {
- for (POSITION pos = m_SelectedCellMap.GetStartPosition(); pos != NULL; )
- {
- DWORD key;
- CCellID cell;
- m_SelectedCellMap.GetNextAssoc(pos, key, (CCellID&)cell);
- m_PrevSelectedCellMap.SetAt(key, cell);
- }
- }
-
- if (m_LeftClickDownCell.row < GetFixedRowCount())
- OnFixedRowClick(m_LeftClickDownCell);
- else if (m_LeftClickDownCell.col < GetFixedColumnCount())
- OnFixedColumnClick(m_LeftClickDownCell);
- else
- {
- m_MouseMode = m_bListMode? MOUSE_SELECT_ROW : MOUSE_SELECT_CELLS;
- OnSelecting(m_LeftClickDownCell);
- }
- m_nTimerID = SetTimer(WM_LBUTTONDOWN, m_nTimerInterval, 0);
- }
- m_LastMousePoint = point;
- }
- void CGridCtrl::OnLButtonUp(UINT nFlags, CPoint point)
- {
- CWnd::OnLButtonUp(nFlags, point);
- #ifndef _WIN32_WCE_NO_CURSOR
- ClipCursor(NULL);
- #endif
- if (GetCapture()->GetSafeHwnd() == GetSafeHwnd())
- {
- ReleaseCapture();
- KillTimer(m_nTimerID);
- m_nTimerID = 0;
- }
- // m_MouseMode == MOUSE_PREPARE_EDIT only if user clicked down on current cell
- // and then didn't move mouse before clicking up (releasing button)
- if (m_MouseMode == MOUSE_PREPARE_EDIT)
- {
- OnEditCell(m_idCurrentCell.row, m_idCurrentCell.col, VK_LBUTTON);
- }
- #ifndef GRIDCONTROL_NO_DRAGDROP
- // m_MouseMode == MOUSE_PREPARE_DRAG only if user clicked down on a selected cell
- // and then didn't move mouse before clicking up (releasing button)
- else if (m_MouseMode == MOUSE_PREPARE_DRAG)
- {
- ResetSelectedRange();
- }
- #endif
- else if (m_MouseMode == MOUSE_SIZING_COL)
- {
- CRect rect;
- GetClientRect(rect);
- CRect invertedRect(m_LastMousePoint.x, rect.top, m_LastMousePoint.x + 2, rect.bottom);
- CDC* pDC = GetDC();
- if (pDC) {
- pDC->InvertRect(&invertedRect);
- ReleaseDC(pDC);
- }
- if (m_LeftClickDownPoint != point)
- {
- CPoint start;
- if (!GetCellOrigin(m_LeftClickDownCell, &start)) return;
- SetColumnWidth(m_LeftClickDownCell.col, point.x - start.x);
- ResetScrollBars();
- Invalidate();
- }
- }
- else if (m_MouseMode == MOUSE_SIZING_ROW)
- {
- CRect rect;
- GetClientRect(rect);
- CRect invertedRect(rect.left, m_LastMousePoint.y, rect.right, m_LastMousePoint.y + 2);
-
- CDC* pDC = GetDC();
- if (pDC) {
- pDC->InvertRect(&invertedRect);
- ReleaseDC(pDC);
- }
-
- if (m_LeftClickDownPoint != point)
- {
- CPoint start;
- if (!GetCellOrigin(m_LeftClickDownCell, &start)) return;
- SetRowHeight(m_LeftClickDownCell.row, point.y - start.y);
- ResetScrollBars();
- Invalidate();
- }
- }
- m_MouseMode = MOUSE_NOTHING;
- #ifndef _WIN32_WCE_NO_CURSOR
- SetCursor(AfxGetApp()->LoadStandardCursor(IDC_ARROW));
- #endif
- if (!IsValid(m_LeftClickDownCell)) return;
- CWnd *pOwner = GetOwner();
- if (pOwner && IsWindow(pOwner->m_hWnd))
- pOwner->PostMessage(WM_COMMAND, MAKELONG(GetDlgCtrlID(), BN_CLICKED),
- (LPARAM) GetSafeHwnd());
- }
- #if !defined(_WIN32_WCE_NO_PRINTING) && !defined(GRIDCONTROL_NO_PRINTING)
- /////////////////////////////////////////////////////////////////////////////
- // CGridCtrl printing
- void CGridCtrl::Print()
- {
- CDC dc;
- CPrintDialog printDlg(FALSE);
- if (printDlg.DoModal() != IDOK) // Get printer settings from user
- return;
- dc.Attach(printDlg.GetPrinterDC()); // attach a printer DC
- dc.m_bPrinting = TRUE;
- CString strTitle;
- strTitle.LoadString(AFX_IDS_APP_TITLE);
- DOCINFO di; // Initialise print doc details
- memset(&di, 0, sizeof (DOCINFO));
- di.cbSize = sizeof (DOCINFO);
- di.lpszDocName = strTitle;
- BOOL bPrintingOK = dc.StartDoc(&di); // Begin a new print job
- CPrintInfo Info;
- Info.m_rectDraw.SetRect(0,0, dc.GetDeviceCaps(HORZRES), dc.GetDeviceCaps(VERTRES));
- OnBeginPrinting(&dc, &Info); // Initialise printing
- for (UINT page = Info.GetMinPage(); page <= Info.GetMaxPage() && bPrintingOK; page++)
- {
- dc.StartPage(); // begin new page
- Info.m_nCurPage = page;
- OnPrint(&dc, &Info); // Print page
- bPrintingOK = (dc.EndPage() > 0); // end page
- }
- OnEndPrinting(&dc, &Info); // Clean up after printing
- if (bPrintingOK)
- dc.EndDoc(); // end a print job
- else
- dc.AbortDoc(); // abort job.
- dc.Detach(); // detach the printer DC
- }
- /////////////////////////////////////////////////////////////////////////////
- // CGridCtrl printing overridables - for Doc/View print/print preview framework
- void CGridCtrl::OnBeginPrinting(CDC *pDC, CPrintInfo *pInfo)
- {
- // OnBeginPrinting() is called after the user has committed to
- // printing by OK'ing the Print dialog, and after the framework
- // has created a CDC object for the printer or the preview view.
- // This is the right opportunity to set up the page range.
- // Given the CDC object, we can determine how many rows will
- // fit on a page, so we can in turn determine how many printed
- // pages represent the entire document.
- ASSERT(pDC && pInfo);
- if (!pDC || !pInfo) return;
- int nMaxRowCount = GetRowCount() - GetFixedRowCount();
- if (!nMaxRowCount) return;
- // Get a DC for the current window (will be a screen DC for print previewing)
- CDC *pCurrentDC = GetDC(); // will have dimensions of the client area
- if (!pCurrentDC) return;
- CSize PaperPixelsPerInch(pDC->GetDeviceCaps(LOGPIXELSX), pDC->GetDeviceCaps(LOGPIXELSY));
- CSize ScreenPixelsPerInch(pCurrentDC->GetDeviceCaps(LOGPIXELSX), pCurrentDC->GetDeviceCaps(LOGPIXELSY));
- // Create the printer font
- int nFontSize = -9;
- CString strFontName = "Times New Roman";
- m_PrinterFont.CreateFont(nFontSize, 0,0,0, FW_NORMAL, 0,0,0, DEFAULT_CHARSET,
- OUT_CHARACTER_PRECIS, CLIP_CHARACTER_PRECIS, DEFAULT_QUALITY,
- DEFAULT_PITCH | FF_DONTCARE, strFontName);
-
- CFont *pOldFont = pDC->SelectObject(&m_PrinterFont);
- // Get the average character width (in GridCtrl units) and hence the margins
- m_CharSize = pDC->GetTextExtent(_T("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSATUVWXYZ"),52);
- m_CharSize.cx /= 52;
- int nMargins = (LEFT_MARGIN+RIGHT_MARGIN)*m_CharSize.cx;
- // Get the page sizes (physical and logical)
- m_PaperSize = CSize(pDC->GetDeviceCaps(HORZRES), pDC->GetDeviceCaps(VERTRES));
- m_LogicalPageSize.cx = GetVirtualWidth()+nMargins;
- #ifdef _WIN32_WCE
- m_LogicalPageSize.cy = (m_LogicalPageSize.cx * m_PaperSize.cy) / m_PaperSize.cx;
- #else
- m_LogicalPageSize.cy = MulDiv(m_LogicalPageSize.cx, m_PaperSize.cy, m_PaperSize.cx);
- #endif
- m_nPageHeight = m_LogicalPageSize.cy - GetFixedRowHeight()
- - (HEADER_HEIGHT+FOOTER_HEIGHT + 2*GAP)*m_CharSize.cy;
- // Get the number of pages. Assumes no row is bigger than the page size.
- int nTotalRowHeight = 0;
- int nNumPages = 1;
- for (int row = GetFixedRowCount(); row < GetRowCount(); row++)
- {
- nTotalRowHeight += GetRowHeight(row);
- if (nTotalRowHeight > m_nPageHeight) {
- nNumPages++;
- nTotalRowHeight = GetRowHeight(row);
- }
- }
- // Set up the print info
- pInfo->SetMaxPage(nNumPages);
- pInfo->m_nCurPage = 1; // start printing at page# 1
- ReleaseDC(pCurrentDC);
- pDC->SelectObject(pOldFont);
- }
- void CGridCtrl::OnPrint(CDC *pDC, CPrintInfo *pInfo)
- {
- if (!pDC || !pInfo) return;
- //CRect rcPage(pInfo->m_rectDraw);
- CFont *pOldFont = pDC->SelectObject(&m_PrinterFont);
- // Set the page map mode to use GridCtrl units, and setup margin
- pDC->SetMapMode(MM_ANISOTROPIC);
- pDC->SetWindowExt(m_LogicalPageSize);
- pDC->SetViewportExt(m_PaperSize);
- pDC->SetWindowOrg(-LEFT_MARGIN*m_CharSize.cx, 0);
- // Header
- pInfo->m_rectDraw.top = 0;
- pInfo->m_rectDraw.left = 0;
- pInfo->m_rectDraw.right = m_LogicalPageSize.cx - (LEFT_MARGIN+RIGHT_MARGIN)*m_CharSize.cx;
- pInfo->m_rectDraw.bottom = HEADER_HEIGHT*m_CharSize.cy;
- PrintHeader(pDC, pInfo);
- pDC->OffsetWindowOrg(0, -HEADER_HEIGHT*m_CharSize.cy);
- // Gap between header and column headings
- pDC->OffsetWindowOrg(0, -GAP*m_CharSize.cy);
- // Print the column headings
- pInfo->m_rectDraw.bottom = GetFixedRowHeight();
- PrintColumnHeadings(pDC, pInfo);
- pDC->OffsetWindowOrg(0, -GetFixedRowHeight());
- // We need to find out which row to start printing for this page.
- int nTotalRowHeight = 0;
- UINT nNumPages = 1;
- int nCurrPrintRow = GetFixedRowCount();
- while (nCurrPrintRow < GetRowCount() && nNumPages < pInfo->m_nCurPage)
- {
- nTotalRowHeight += GetRowHeight(nCurrPrintRow);
- if (nTotalRowHeight > m_nPageHeight) {
- nNumPages++;
- if (nNumPages == pInfo->m_nCurPage) break;
- nTotalRowHeight = GetRowHeight(nCurrPrintRow);
- }
- nCurrPrintRow++;
- }
- if (nCurrPrintRow >= GetRowCount()) return;
- // Draw as many rows as will fit on the printed page.
- // Clip the printed page so that there is no partially shown
- // row at the bottom of the page (the same row which will be fully
- // shown at the top of the next page).
- BOOL bFirstPrintedRow = TRUE;
- CRect rect;
- rect.bottom = -1;
- while (nCurrPrintRow < GetRowCount())
- {
- rect.top = rect.bottom+1;
- rect.bottom = rect.top + GetRowHeight(nCurrPrintRow) - 1;
- if (rect.bottom > m_nPageHeight) break; // Gone past end of page
- rect.right = -1;
- for (int col = 0; col < GetColumnCount(); col++)
- {
- rect.left = rect.right+1;
- rect.right = rect.left + GetColumnWidth(col) - 1;
- DrawCell(pDC, nCurrPrintRow, col, rect);
- if (m_nGridLines == GVL_BOTH || m_nGridLines == GVL_HORZ)
- {
- int Overlap = (col == 0)? 0:1;
- pDC->MoveTo(rect.left-Overlap, rect.bottom);
- pDC->LineTo(rect.right, rect.bottom);
- if (nCurrPrintRow == 0) {
- pDC->MoveTo(rect.left-Overlap, rect.top);
- pDC->LineTo(rect.right, rect.top);
- }
- }
- if (m_nGridLines == GVL_BOTH || m_nGridLines == GVL_VERT)
- {
- int Overlap = (bFirstPrintedRow)? 0:1;
- pDC->MoveTo(rect.right, rect.top-Overlap);
- pDC->LineTo(rect.right, rect.bottom);
- if (col == 0) {
- pDC->MoveTo(rect.left, rect.top-Overlap);
- pDC->LineTo(rect.left, rect.bottom);
- }
- }
- }
- nCurrPrintRow++;
- bFirstPrintedRow = FALSE;
- }
- // Footer
- pInfo->m_rectDraw.bottom = FOOTER_HEIGHT*m_CharSize.cy;
- pDC->SetWindowOrg(-LEFT_MARGIN*m_CharSize.cx, -m_LogicalPageSize.cy + FOOTER_HEIGHT*m_CharSize.cy);
- PrintFooter(pDC, pInfo);
- // SetWindowOrg back for next page
- pDC->SetWindowOrg(0,0);
- pDC->SelectObject(pOldFont);
- }
- void CGridCtrl::PrintColumnHeadings(CDC *pDC, CPrintInfo* /*pInfo*/)
- {
- CFont *pOldFont = pDC->SelectObject(&m_PrinterFont);
- CRect rect;
- rect.bottom = -1;
- for (int row = 0; row < GetFixedRowCount(); row++)
- {
- rect.top = rect.bottom+1;
- rect.bottom = rect.top + GetRowHeight(row) - 1;
- rect.right = -1;
- for (int col = 0; col < GetColumnCount(); col++)
- {
- rect.left = rect.right+1;
- rect.right = rect.left + GetColumnWidth(col) - 1;
- DrawFixedCell(pDC, row, col, rect);
- if (m_nGridLines == GVL_BOTH || m_nGridLines == GVL_HORZ)
- {
- int Overlap = (col == 0)? 0:1;
- pDC->MoveTo(rect.left-Overlap, rect.bottom);
- pDC->LineTo(rect.right, rect.bottom);
- if (row == 0) {
- pDC->MoveTo(rect.left-Overlap, rect.top);
- pDC->LineTo(rect.right, rect.top);
- }
- }
- if (m_nGridLines == GVL_BOTH || m_nGridLines == GVL_VERT)
- {
- int Overlap = (row == 0)? 0:1;
- pDC->MoveTo(rect.right, rect.top-Overlap);
- pDC->LineTo(rect.right, rect.bottom);
- if (col == 0) {
- pDC->MoveTo(rect.left, rect.top-Overlap);
- pDC->LineTo(rect.left, rect.bottom);
- }
- }
- }
- }
- pDC->SelectObject(pOldFont);
- }
- void CGridCtrl::PrintHeader(CDC *pDC, CPrintInfo *pInfo)
- {
- CRect rc(pInfo->m_rectDraw);
- CString strHeaderString;
- CFont BoldFont;
- LOGFONT lf;
- //create bold font for header and footer
- VERIFY(m_PrinterFont.GetLogFont(&lf));
- lf.lfWeight = FW_BOLD;
- VERIFY(BoldFont.CreateFontIndirect(&lf));
-
- CFont *pNormalFont = pDC->SelectObject(&BoldFont);
- int nPrevBkMode = pDC->SetBkMode(TRANSPARENT);
- // print App title on top right margin
- strHeaderString.LoadString(AFX_IDS_APP_TITLE);
- pDC->DrawText(strHeaderString, &rc, DT_RIGHT | DT_SINGLELINE | DT_NOPREFIX | DT_VCENTER);
- // print parent window title in the centre (Gert Rijs)
- CWnd *pParentWnd = GetParent();
- while (pParentWnd)
- {
- pParentWnd->GetWindowText(strHeaderString);
- if (strHeaderString.GetLength()) // can happen if it is a CView, CChildFrm has the title
- break;
- pParentWnd = pParentWnd->GetParent();
- }
- pDC->DrawText(strHeaderString, &rc, DT_CENTER | DT_SINGLELINE | DT_NOPREFIX | DT_VCENTER);
- pDC->SetBkMode(nPrevBkMode);
- pDC->SelectObject(pNormalFont);
- BoldFont.DeleteObject();
- pDC->SelectStockObject(BLACK_PEN);
- pDC->MoveTo(rc.left, rc.bottom);
- pDC->LineTo(rc.right, rc.bottom);
- }
- //print footer with a line and date, and page number
- void CGridCtrl::PrintFooter(CDC *pDC, CPrintInfo *pInfo)
- {
- CRect rc(pInfo->m_rectDraw);
- CFont BoldFont;
- LOGFONT lf;
- //draw line
- pDC->MoveTo(rc.left, rc.top);
- pDC->LineTo(rc.right, rc.top);
- //create bold font for header and footer
- m_PrinterFont.GetLogFont(&lf);
- lf.lfWeight = FW_BOLD;
- BoldFont.CreateFontIndirect(&lf);
- CFont *pNormalFont = pDC->SelectObject(&BoldFont);
- int nPrevBkMode = pDC->SetBkMode(TRANSPARENT);
- // draw page number
- CString sTemp ;
- rc.OffsetRect(0, m_CharSize.cy/2);
- sTemp.Format(_T("Page %d of %d"), pInfo->m_nCurPage, pInfo->GetMaxPage());
- pDC->DrawText(sTemp,-1,rc, DT_LEFT | DT_SINGLELINE | DT_NOPREFIX | DT_NOCLIP | DT_VCENTER);
- CTime t = CTime::GetCurrentTime();
- sTemp = t.Format(_T("%c"));
- pDC->DrawText(sTemp,-1,rc, DT_RIGHT | DT_SINGLELINE | DT_NOPREFIX | DT_NOCLIP | DT_VCENTER);
- pDC->SetBkMode(nPrevBkMode);
- pDC->SelectObject(pNormalFont);
- BoldFont.DeleteObject();
- }
- void CGridCtrl::OnEndPrinting(CDC* /*pDC*/, CPrintInfo* /*pInfo*/)
- {
- m_PrinterFont.DeleteObject();
- }
- #endif // !defined(_WIN32_WCE_NO_PRINTING) && !defined(GRIDCONTROL_NO_PRINTING)
- #ifndef _WIN32_WCE
- /////////////////////////////////////////////////////////////////////////////
- // CGridCtrl persistance
- BOOL CGridCtrl::Save(LPCTSTR filename)
- {
- CStdioFile File;
- CFileException ex;
- if (!File.Open(filename, CFile::modeWrite | CFile::modeCreate| CFile::typeText, &ex)) {
- ex.ReportError();
- return FALSE;
- }
- TRY {
- int nNumColumns = GetColumnCount();
- for (int i = 0; i < nNumColumns; i++) {
- File.WriteString(GetItemText(0,i));
- File.WriteString((i==(nNumColumns-1))? _T("\n"):_T(","));
- }
- for (int i = 0; i < GetRowCount(); i++) {
- for (int j = 0; j < nNumColumns; j++) {
- File.WriteString(GetItemText(i,j));
- File.WriteString((j==(nNumColumns-1))? _T("\n"):_T(","));
- }
- }
- File.Close();
- }
- CATCH (CFileException, e) {
- AfxMessageBox(_T("Unable to save grid list"));
- e->Delete();
- return FALSE;
- }
- END_CATCH
- return TRUE;
- }
- BOOL CGridCtrl::Load(LPCTSTR filename)
- {
- TCHAR *token, *end;
- TCHAR buffer[1024];
- CStdioFile File;
- CFileException ex;
- if (!File.Open(filename, CFile::modeRead | CFile::typeText)) {
- ex.ReportError();
- return FALSE;
- }
- DeleteAllItems();
- TRY {
- // Read Header off file
- File.ReadString(buffer, 1024);
- // Get first token
- for (token=buffer, end=buffer;
- *end && (*end != _T(',')) && (*end != _T('\n')); end++);
- if ((*end == _T('\0')) && (token == end)) token = NULL;
- *end = _T('\0');
- while (token)
- {
- InsertColumn(token);
-
- // Get next token
- for (token=++end; *end && (*end != _T(',')) && (*end != _T('\n'));
- end++);
- if ((*end == _T('\0')) && (token == end)) token = NULL;
- *end = _T('\0');
- }
- // Read in rest of data
- int nItem = 0;
- while (File.ReadString(buffer, 1024)) {
- // Get first token
- for (token=buffer, end=buffer;
- *end && (*end != _T(',')) && (*end != _T('\n')); end++);
- if ((*end == _T('\0')) && (token == end)) token = NULL;
- *end = _T('\0');
- int nSubItem = 0;
- while (token) {
- if (!nSubItem)
- InsertRow(token);
- else
- SetItemText(nItem, nSubItem, token);
- // Get next token
- for (token=++end; *end && (*end != _T(',')) && (*end != _T('\n'));
- end++);
- if ((*end == _T('\0')) && (token == end)) token = NULL;
- *end = _T('\0');
- nSubItem++;
- }
- nItem++;
- }
- AutoSizeColumns();
- File.Close();
- }
- CATCH (CFileException, e) {
- AfxMessageBox(_T("Unable to load grid data"));
- e->Delete();
- return FALSE;
- }
- END_CATCH
- return TRUE;
- }
- #endif
- /////////////////////////////////////////////////////////////////////////////
- // CGridCtrl overrideables
- #ifndef GRIDCONTROL_NO_DRAGDROP
- // This is no longer needed since I've changed to OLE drag and drop - but it's
- // still cool code. :)
- CImageList* CGridCtrl::CreateDragImage(CPoint *pHotSpot)
- {
- CDC* pDC = GetDC();
- if (!pDC) return NULL;
- CRect rect;
- CCellID cell = GetFocusCell();
- if (!GetCellRect(cell.row, cell.col, rect)) return NULL;
- // Translate coordinate system
- rect.BottomRight() = CPoint(rect.Width(), rect.Height());
- rect.TopLeft() = CPoint(0,0);
- *pHotSpot = rect.BottomRight();
- // Create a new imagelist (the caller of this function has responsibility
- // for deleting this list)
- CImageList* pList = new CImageList;
- if (!pList || !pList->Create(rect.Width(), rect.Height(), ILC_MASK, 1,1))
- {
- if (pList) delete pList;
- return NULL;
- }
- // Create mem DC and bitmap
- CDC MemDC;
- CBitmap bm;
- MemDC.CreateCompatibleDC(pDC);
- bm.CreateCompatibleBitmap(pDC, rect.Width(), rect.Height());
- CBitmap* pOldBitmap = MemDC.SelectObject(&bm);
- MemDC.SetWindowOrg(0,0);
- // Draw cell onto bitmap in memDC
- DrawCell(&MemDC, cell.row, cell.col, rect, TRUE);
- // Clean up
- MemDC.SelectObject(pOldBitmap);
- ReleaseDC(pDC);
- // Add the bitmap we just drew to the image list.
- pList->Add(&bm, GetTextBkColor());
- bm.DeleteObject();
- return pList;
- }
- #endif
- void CGridCtrl::OnFixedRowClick(CCellID& cell)
- {
- if (!IsValid(cell))
- return;
- if (m_bListMode)
- {
- if (!m_bSortOnClick)
- return;
- CWaitCursor waiter;
- if (cell.col == m_SortColumn)
- m_bAscending = !m_bAscending;
- else
- {
- m_bAscending = TRUE;
- m_SortColumn = cell.col;
- }
- SortTextItems(m_SortColumn, m_bAscending);
- Invalidate();
- }
- else if (cell.col < GetFixedColumnCount())
- {
- m_MouseMode = MOUSE_SELECT_ALL;
- OnSelecting(cell);
- }
- else
- {
- m_MouseMode = MOUSE_SELECT_COL;
- OnSelecting(cell);
- }
- }
- void CGridCtrl::OnFixedColumnClick(CCellID& cell)
- {
- if (!IsValid(cell))
- return;
- // if (m_bListMode && (GetItemState(cell.row, m_nFixedCols) & GVNI_SELECTED))
- // {
- // OnEditCell(cell.row, cell.col, VK_LBUTTON);
- // return;
- // }
- if (cell.row < GetFixedRowCount())
- {
- m_MouseMode = MOUSE_SELECT_ALL;
- OnSelecting(cell);
- }
- else
- {
- m_MouseMode = MOUSE_SELECT_ROW;
- OnSelecting(cell);
- }
- }
- // Gets the extent of the text pointed to by str (no CDC needed)
- // By default this uses the selected font (which is a bigger font)
- CSize CGridCtrl::GetTextExtent(LPCTSTR str, BOOL bUseSelectedFont /* = TRUE */)
- {
- CDC* pDC = GetDC();
- if (!pDC) return CSize(0,0);
- CFont *pOldFont, font;
- if (bUseSelectedFont)
- {
- LOGFONT lf;
- memcpy(&lf, &m_Logfont, sizeof(LOGFONT));
- lf.lfWeight = SELECTED_CELL_FONT_WEIGHT;
-
- font.CreateFontIndirect(&lf);
-
- pOldFont = pDC->SelectObject(&font);
- }
- else
- pOldFont = pDC->SelectObject(&m_Font);
- CSize size = pDC->GetTextExtent(str);
- pDC->SelectObject(pOldFont);
- ReleaseDC(pDC);
- return size + CSize(2*m_nMargin, 2*m_nMargin);
- }
- CSize CGridCtrl::GetCellExtent(int nRow, int nCol, CDC* pDC)
- {
- LOGFONT *pLF = GetItemFont(nRow, nCol);
-
- // use selected font since it's thicker
- LOGFONT lf;
- memcpy(&lf, pLF, sizeof(LOGFONT));
-
- if (nRow < m_nFixedRows || nCol < m_nFixedCols)
- lf.lfWeight = SELECTED_CELL_FONT_WEIGHT;
-
- CFont font;
- font.CreateFontIndirect(&lf);
-
- CFont* pOldFont = pDC->SelectObject(&font);
- CSize size = pDC->GetTextExtent(GetItemText(nRow, nCol));
- pDC->SelectObject(pOldFont);
- size += CSize(4*m_nMargin, 2*m_nMargin);
- CSize ImageSize(0,0);
- if (m_pImageList) {
- int nImage = GetItemImage(nRow, nCol);
- if (nImage >= 0) {
- IMAGEINFO Info;
- if (m_pImageList->GetImageInfo(nImage, &Info))
- ImageSize = CSize(Info.rcImage.right-Info.rcImage.left+1,
- Info.rcImage.bottom-Info.rcImage.top+1);
- }
- }
-
- return CSize(size.cx + ImageSize.cx, max(size.cy, ImageSize.cy));
- }
- BOOL CGridCtrl::DrawFixedCell(CDC* pDC, int nRow, int nCol, CRect rect, BOOL bEraseBk)
- {
- if (!m_bAllowDraw)
- return FALSE;
- GV_ITEM Item;
- Item.mask = GVIF_TEXT | GVIF_FORMAT | GVIF_IMAGE | GVIF_BKCLR | GVIF_FGCLR;
- Item.row = nRow;
- Item.col = nCol;
- if (!GetItem(&Item))
- return FALSE;
- // Force redraw of background if custom colour
- //if (Item.crBkClr != CLR_DEFAULT)
- bEraseBk = TRUE;
- if (bEraseBk
- #if !defined(_WIN32_WCE_NO_PRINTING) && !defined(GRIDCONTROL_NO_PRINTING)
- && !pDC->IsPrinting()
- #endif
- )
- {
- CBrush brush((Item.crBkClr == CLR_DEFAULT)? GetFixedBkColor() : Item.crBkClr);
- pDC->FillRect(rect, &brush);
- }
- pDC->SetTextColor((Item.crFgClr == CLR_DEFAULT)? GetFixedTextColor() : Item.crFgClr);
-
- int nSavedDC = pDC->SaveDC();
-
- // Create the appropriate font and select into DC
- LOGFONT *pLF = GetItemFont(nRow, nCol);
- CCellID FocusCell = GetFocusCell();
- // if (IsCellFixed(nRow, nCol) && (FocusCell.row == nRow || FocusCell.col == nCol))
- //{
- if (FocusCell.row == nRow || FocusCell.col == nCol)
- {
- static LOGFONT lf;
- memcpy(&lf, pLF, sizeof(LOGFONT));
- lf.lfWeight = SELECTED_CELL_FONT_WEIGHT;
- pLF = &lf;
- }
-
- CFont Font;
- Font.CreateFontIndirect(pLF);
- pDC->SelectObject(&Font);
- if (IsValid(FocusCell) && (FocusCell.row == nRow || FocusCell.col == nCol))
- {
- rect.right++; rect.bottom++;
- pDC->DrawEdge(rect, BDR_SUNKENINNER /*EDGE_RAISED*/, BF_RECT);
- rect.DeflateRect(1,1);
- }
- else
- {
- CPen lightpen(PS_SOLID, 1, ::GetSysColor(COLOR_3DHIGHLIGHT)),
- darkpen(PS_SOLID, 1, ::GetSysColor(COLOR_3DDKSHADOW)),
- *pOldPen = pDC->GetCurrentPen();
-
- pDC->SelectObject(&lightpen);
- pDC->MoveTo(rect.right, rect.top);
- pDC->LineTo(rect.left, rect.top);
- pDC->LineTo(rect.left, rect.bottom);
- pDC->SelectObject(&darkpen);
- pDC->MoveTo(rect.right, rect.top);
- pDC->LineTo(rect.right, rect.bottom);
- pDC->LineTo(rect.left, rect.bottom);
- pDC->SelectObject(pOldPen);
- rect.DeflateRect(1,1);
- }
- pDC->SetBkMode(TRANSPARENT);
- rect.DeflateRect(m_nMargin, 0);
- if (m_pImageList && Item.iImage >= 0)
- {
- IMAGEINFO Info;
- if (m_pImageList->GetImageInfo(Item.iImage, &Info))
- {
- //CRgn rgn;
- //rgn.CreateRectRgnIndirect(rect);
- //pDC->SelectClipRgn(&rgn);
- //rgn.DeleteObject();
-
- int nImageWidth = Info.rcImage.right-Info.rcImage.left+1;
- m_pImageList->Draw(pDC, Item.iImage, rect.TopLeft(), ILD_NORMAL);
- rect.left += nImageWidth+m_nMargin;
- }
- }
- // rect.bottom += 10;
- // Item.nFormat = DT_LEFT|DT_VCENTER;
- DrawText(pDC->m_hDC, Item.szText, -1, rect, Item.nFormat);
- pDC->RestoreDC(nSavedDC);
- Font.DeleteObject();
- return TRUE;
- }
- BOOL CGridCtrl::DrawCell(CDC* pDC, int nRow, int nCol, CRect rect, BOOL bEraseBk)
- {
- if (!m_bAllowDraw)
- return FALSE;
- GV_ITEM Item;
- Item.mask = GVIF_TEXT | GVIF_FORMAT | GVIF_STATE | GVIF_IMAGE | GVIF_BKCLR | GVIF_FGCLR;
- Item.row = nRow;
- Item.col = nCol;
- if (!GetItem(&Item))
- return FALSE;
- COLORREF TextBkClr = (Item.crBkClr == CLR_DEFAULT)? GetTextBkColor() : Item.crBkClr;
- COLORREF TextClr = (Item.crFgClr == CLR_DEFAULT)? GetTextColor() : Item.crFgClr;
- // Force redraw of background if custom colour
- //if (Item.crBkClr != CLR_DEFAULT)
- bEraseBk = TRUE;
- int nSavedDC = pDC->SaveDC();
- pDC->SetBkMode(TRANSPARENT);
- if (Item.state & GVIS_FOCUSED
- #if !defined(_WIN32_WCE_NO_PRINTING) && !defined(GRIDCONTROL_NO_PRINTING)
- && !pDC->IsPrinting()
- #endif
- )
- {
- rect.right++; rect.bottom++; // FillRect doesn't draw RHS or bottom
- if (bEraseBk)
- {
- CBrush brush(TextBkClr);
- pDC->FillRect(rect, &brush);
- }
- rect.right--; rect.bottom--;
- pDC->SelectStockObject(BLACK_PEN);
- pDC->SelectStockObject(NULL_BRUSH);
- pDC->Rectangle(rect);
- pDC->SetTextColor(TextClr);
- rect.DeflateRect(1,1);
- }
- else if (Item.state & GVIS_SELECTED
- #if !defined(_WIN32_WCE_NO_PRINTING) && !defined(GRIDCONTROL_NO_PRINTING)
- && !pDC->IsPrinting()
- #endif
- )
- {
- rect.right++; rect.bottom++; // FillRect doesn't draw RHS or bottom
- pDC->FillSolidRect(rect, ::GetSysColor(COLOR_HIGHLIGHT));
- rect.right--; rect.bottom--;
- pDC->SetTextColor(::GetSysColor(COLOR_HIGHLIGHTTEXT));
- } else {
- rect.right++; rect.bottom++; // FillRect doesn't draw RHS or bottom
- if (bEraseBk)
- {
- CBrush brush(TextBkClr);
- pDC->FillRect(rect, &brush);
- }
- rect.right--; rect.bottom--;
- pDC->SetTextColor(TextClr);
- }
- if (Item.state & GVIS_DROPHILITED
- #if !defined(_WIN32_WCE_NO_PRINTING) && !defined(GRIDCONTROL_NO_PRINTING)
- && !pDC->IsPrinting()
- #endif
- )
- {
- pDC->SelectStockObject(BLACK_PEN);
- pDC->SelectStockObject(NULL_BRUSH);
- pDC->Rectangle(rect);
- }
- // Create the appropriate font and select into DC
- LOGFONT *pLF = GetItemFont(nRow, nCol);
- //CCellID FocusCell = GetFocusCell();
- //if (IsCellFixed(nRow, nCol) && (FocusCell.row == nRow || FocusCell.col == nCol))
- //{
- // static LOGFONT lf;
- // memcpy(&lf, pLF, sizeof(LOGFONT));
- // lf.lfWeight = SELECTED_CELL_FONT_WEIGHT;
- // pLF = &lf;
- //}
-
- CFont Font;
- Font.CreateFontIndirect(pLF);
- pDC->SelectObject(&Font);
- rect.DeflateRect(m_nMargin, 0);
- if (m_pImageList && Item.iImage >= 0)
- {
- IMAGEINFO Info;
- if (m_pImageList->GetImageInfo(Item.iImage, &Info))
- {
- //CRgn rgn;
- //rgn.CreateRectRgnIndirect(rect);
- //pDC->SelectClipRgn(&rgn);
- //rgn.DeleteObject();
- int nImageWidth = Info.rcImage.right-Info.rcImage.left+1;
- m_pImageList->Draw(pDC, Item.iImage, rect.TopLeft(), ILD_NORMAL);
- rect.left += nImageWidth+m_nMargin;
- }
- }
- DrawText(pDC->m_hDC, Item.szText, -1, rect, Item.nFormat);
- pDC->RestoreDC(nSavedDC);
- Font.DeleteObject();
- return TRUE;
- }
- void CGridCtrl::OnEditCell(int nRow, int nCol, UINT nChar)
- {
- EnsureVisible(nRow, nCol);
- CCellID cell(nRow, nCol);
- if (!IsValid(cell) || !IsCellEditable(nRow, nCol) || !IsCellVisible(nRow, nCol))
- return;
- bEditing = true;
- CRect rect;
- if (!GetCellRect(cell, rect)) return;
- SendMessageToParent(nRow, nCol, GVN_BEGINLABELEDIT);
- GV_ITEM Item;
- Item.mask = GVIF_TEXT | GVIF_FORMAT;
- Item.row = nRow;
- Item.col = nCol;
- if (!GetItem(&Item)) return;
- DWORD dwStyle = ES_LEFT;
- if (Item.nFormat & DT_RIGHT) dwStyle = ES_RIGHT;
- else if (Item.nFormat & DT_CENTER) dwStyle = ES_CENTER;
- CreateInPlaceEditControl(rect, dwStyle, IDC_INPLACE_CONTROL,
- nRow, nCol, Item.szText, nChar);
- }
- void CGridCtrl::CreateInPlaceEditControl(CRect& rect, DWORD dwStyle, UINT nID,
- int nRow, int nCol,
- LPCTSTR szText, int nChar)
- {
- // InPlaceEdit auto-deletes itself
- new CInPlaceEdit(this, rect, dwStyle, nID, nRow, nCol, szText, nChar);
- }
- void CGridCtrl::OnEndEditCell(int nRow, int nCol, CString str)
- {
- CString strCurrent = GetItemText(nRow,nCol);
- if (strCurrent != str)
- {
- SetModified(TRUE, nRow, nCol);
- SetItemText(nRow, nCol, str);
- }
- if( nParentType == 1 )
- {
- CModeDlg* pModeDlg = ((CModeDlg*)GetParent());
- pModeDlg->GridVerifyData();
- bEditing = false;
- }
- else if( nParentType == 2 )
- {
- CReliefDlg* pReliefDlg = ((CReliefDlg*)GetParent());
- pReliefDlg->GridVerifyData();
- bEditing = false;
- }
- }
- CString CGridCtrl::GetItemText(int nRow, int nCol)
- {
- if (nRow < 0 || nRow >= m_nRows || nCol < 0 || nCol >= m_nCols) return "";
- CGridCell* pCell = GetCell(nRow, nCol);
- ASSERT(pCell);
- if (!pCell) return "";
- return pCell->szText;
- }
|