2013年10月25日 星期五

[W8App] 8.0 “Snapped” to 8.1– with LayoutAwarePage

Windows 8.1的一個德政是取消了原本的Snappped View限制,App除了FullScreen外還可以有任意的大小(只是有最小限制)。不過對於已經開發好的8.0 App,就要花時間重新調整不同視窗大小時Layout。

我的懶人改法是,除了FullScreen外一律視為Snapped。依這樣的邏輯其實只要改動一個function – LayoutAwarePage的DetermineVisualState:

   1: protected virtual string DetermineVisualState(ApplicationView appView)
   2: {
   3:     if(appView.IsFullScreen)
   4:     {
   5:         return @"FullScreen" + appView.Orientation.ToString();
   6:     }
   7:     else
   8:     {
   9:         return @"Snapped";
  10:     }
  11: }

1. 原本參數為ApplicationViewState (編譯時有warning),改為ApplicationView
3-10. 原本view state可以直接吐回字串的部份用邏輯修正


當然呼叫端也要調整為



   1: string visualState = DetermineVisualState(ApplicationView.GetForCurrentView());
1.  原本呼叫端是傳入ApplicationView.Value (編譯時有warning),改為ApplicationView.GetForCurrentView()

2013年10月22日 星期二

[W8App] In W8.1, DirectWrite must create font with IDWriteFactory2

升上8.1第一天後, 我的App發生DirectWrite繪製文字有時會繪製不出來的問題。最後才發現改IDWriteFactory改為IDWriteFactory2後就一切正常,不過IDWriteFactory2只支援8.1以上,所以結論是App必做二個版本了。。。

2013年10月14日 星期一

[W8App] D2DERR_WRONG_FACTORY (0x88990012)

最近在學習W8App的Direct2D與DirectWrite時發生了一個例外的錯誤,當呼叫D2D的EndDraw來更新繪圖結果是產生了0x88990012 (D2DERR_WRONG_FACTORY)的例外錯誤。

這個錯誤發生的原因是我的DirectWrite Render物件預先儲存了D2D factory,而透過SurfaceImageSource來連結W8App與Direct2D時,D2DContext所使用的D2D factory似乎有自已獨立的產生方式。

目前的解決方式是透過GetFactory直接從D2DContext物件中取得D2DFactory,至於是否有更好的作法,就需要再研究。

   1: IFACEMETHODIMP CustomTextRenderer::DrawGlyphRun(
   2:     _In_opt_ void* clientDrawingContext,
   3:     FLOAT baselineOriginX,
   4:     FLOAT baselineOriginY,
   5:     DWRITE_MEASURING_MODE measuringMode,
   6:     _In_ DWRITE_GLYPH_RUN const* glyphRun,
   7:     _In_ DWRITE_GLYPH_RUN_DESCRIPTION const* glyphRunDescription,
   8:     IUnknown* clientDrawingEffect
   9:     )
  10: {
  11:     HRESULT hr = S_OK;
  12:  
  13:     Microsoft::WRL::ComPtr<ID2D1Factory> d2dFactory;
  14:     m_d2dContext->GetFactory(&d2dFactory);
  15:  
  16:     // Create the path geometry.
  17:     Microsoft::WRL::ComPtr<ID2D1PathGeometry> pathGeometry;
  18:     hr = d2dFactory->CreatePathGeometry(&pathGeometry);
  19:  
  20:     // Write to the path geometry using the geometry sink.
  21:  
  22:     Microsoft::WRL::ComPtr<ID2D1GeometrySink> sink;
  23:     if (SUCCEEDED(hr))
  24:     {
  25:         hr = pathGeometry->Open(&sink);
  26:     }
  27:  
  28:     // Get the glyph run outline geometries back from DirectWrite and place them within the
  29:     // geometry sink.
  30:     if (SUCCEEDED(hr))
  31:     {
  32:         hr = glyphRun->fontFace->GetGlyphRunOutline(
  33:             glyphRun->fontEmSize,
  34:             glyphRun->glyphIndices,
  35:             glyphRun->glyphAdvances,
  36:             glyphRun->glyphOffsets,
  37:             glyphRun->glyphCount,
  38:             glyphRun->isSideways,
  39:             glyphRun->bidiLevel %2,
  40:             sink.Get()
  41:             );
  42:     }
  43:  
  44:     // Close the geometry sink
  45:     if (SUCCEEDED(hr))
  46:     {
  47:         hr = sink.Get()->Close();
  48:     }
  49:  
  50:     // Initialize a matrix to translate the origin of the glyph run.
  51:     D2D1::Matrix3x2F const matrix = D2D1::Matrix3x2F(
  52:         1.0f, 0.0f,
  53:         0.0f, 1.0f,
  54:         baselineOriginX, baselineOriginY
  55:         );
  56:  
  57:     // Create the transformed geometry
  58:     Microsoft::WRL::ComPtr<ID2D1TransformedGeometry> transformedGeometry;
  59:     if (SUCCEEDED(hr))
  60:     {
  61:         hr = d2dFactory->CreateTransformedGeometry(
  62:             pathGeometry.Get(),
  63:             &matrix,
  64:             &transformedGeometry
  65:             );
  66:     }
  67:  
  68:     // Draw the outline of the glyph run
  69:     if(SUCCEEDED(hr))
  70:     {
  71:         m_d2dContext->DrawGeometry(transformedGeometry.Get(),m_brushOutline.Get());
  72:         m_d2dContext->FillGeometry(transformedGeometry.Get(),m_brushFill.Get());
  73:     }
  74:  
  75:     return hr;
  76: }

13-14 透過目前的D2DContext來取得D2DFactory

2013年10月11日 星期五

[W8App] Hide / Show cursor

隱藏滑鼠指標

   1: if (Window.Current.CoreWindow.PointerCursor != null) Window.Current.CoreWindow.PointerCursor = null;

顯示滑鼠指標



   1: if (Window.Current.CoreWindow.PointerCursor == null)
   2: {
   3:     Window.Current.CoreWindow.PointerCursor = new CoreCursor(CoreCursorType.Arrow, 1);
   4: }


CoreCursor的id可以自已隨意的指派並不一定是1。


Reference:



  1. CoreCursorType enumeration