Merge branch 'master' into blender2.8
authorCampbell Barton <ideasman42@gmail.com>
Mon, 18 Sep 2017 13:56:19 +0000 (23:56 +1000)
committerCampbell Barton <ideasman42@gmail.com>
Mon, 18 Sep 2017 13:56:19 +0000 (23:56 +1000)
1  2 
source/blender/blenkernel/intern/customdata.c
source/blender/editors/interface/interface.c
source/blender/makesrna/intern/rna_mesh.c

index 3ccf619fd116a586cd93cf46adece9c85624b5fc,68acb60f21a65385cabd4a8f91799529cb31c7f5..eb35f3daad0654f9db1faf7fb7cb3fb6fbc6d184
@@@ -390,19 -390,37 +390,19 @@@ static void layerSwap_tface(void *data
  {
        MTFace *tf = data;
        float uv[4][2];
 -      static const short pin_flags[4] = { TF_PIN1, TF_PIN2, TF_PIN3, TF_PIN4 };
 -      static const char sel_flags[4] = { TF_SEL1, TF_SEL2, TF_SEL3, TF_SEL4 };
 -      short unwrap = tf->unwrap & ~(TF_PIN1 | TF_PIN2 | TF_PIN3 | TF_PIN4);
 -      char flag = tf->flag & ~(TF_SEL1 | TF_SEL2 | TF_SEL3 | TF_SEL4);
        int j;
  
        for (j = 0; j < 4; ++j) {
                const int source_index = corner_indices[j];
 -
                copy_v2_v2(uv[j], tf->uv[source_index]);
 -
 -              /* swap pinning flags around */
 -              if (tf->unwrap & pin_flags[source_index]) {
 -                      unwrap |= pin_flags[j];
 -              }
 -
 -              /* swap selection flags around */
 -              if (tf->flag & sel_flags[source_index]) {
 -                      flag |= sel_flags[j];
 -              }
        }
  
        memcpy(tf->uv, uv, sizeof(tf->uv));
 -      tf->unwrap = unwrap;
 -      tf->flag = flag;
  }
  
  static void layerDefault_tface(void *data, int count)
  {
 -      static MTFace default_tf = {{{0, 0}, {1, 0}, {1, 1}, {0, 1}}, NULL,
 -                                      0, 0, TF_DYNAMIC | TF_CONVERTED, 0, 0};
 +      static MTFace default_tf = {{{0, 0}, {1, 0}, {1, 1}, {0, 1}}};
        MTFace *tf = (MTFace *)data;
        int i;
  
@@@ -810,10 -828,10 +810,10 @@@ static void layerInterp_mloopcol
         * although weights should also not cause this situation */
  
        /* also delay writing to the destination incase dest is in sources */
-       mc->r = CLAMPIS(iroundf(col.r), 0, 255);
-       mc->g = CLAMPIS(iroundf(col.g), 0, 255);
-       mc->b = CLAMPIS(iroundf(col.b), 0, 255);
-       mc->a = CLAMPIS(iroundf(col.a), 0, 255);
+       mc->r = round_fl_to_uchar_clamp(col.r);
+       mc->g = round_fl_to_uchar_clamp(col.g);
+       mc->b = round_fl_to_uchar_clamp(col.b);
+       mc->a = round_fl_to_uchar_clamp(col.a);
  }
  
  static int layerMaxNum_mloopcol(void)
@@@ -1036,10 -1054,10 +1036,10 @@@ static void layerInterp_mcol
                
                /* Subdivide smooth or fractal can cause problems without clamping
                 * although weights should also not cause this situation */
-               mc[j].a = CLAMPIS(iroundf(col[j].a), 0, 255);
-               mc[j].r = CLAMPIS(iroundf(col[j].r), 0, 255);
-               mc[j].g = CLAMPIS(iroundf(col[j].g), 0, 255);
-               mc[j].b = CLAMPIS(iroundf(col[j].b), 0, 255);
+               mc[j].a = round_fl_to_uchar_clamp(col[j].a);
+               mc[j].r = round_fl_to_uchar_clamp(col[j].r);
+               mc[j].g = round_fl_to_uchar_clamp(col[j].g);
+               mc[j].b = round_fl_to_uchar_clamp(col[j].b);
        }
  }
  
@@@ -1172,15 -1190,6 +1172,15 @@@ static void layerSwap_flnor(void *data
        memcpy(flnors, nors, sizeof(nors));
  }
  
 +static void layerDefault_fmap(void *data, int count)
 +{
 +      int *fmap_num = (int *)data;
 +      int i;
 +      for (i = 0; i < count; i++) {
 +              *fmap_num = -1;
 +      }
 +}
 +
  static const LayerTypeInfo LAYERTYPEINFO[CD_NUMTYPES] = {
        /* 0: CD_MVERT */
        {sizeof(MVert), "MVert", 1, NULL, NULL, NULL, NULL, NULL, NULL},
        /* 3 floats per normal vector */
        {sizeof(float) * 3, "vec3f", 1, NULL, NULL, NULL, layerInterp_normal, NULL, NULL,
         NULL, NULL, NULL, NULL, NULL, layerCopyValue_normal},
 -      /* 9: CD_POLYINDEX */  /* DEPRECATED */
 -      {sizeof(int), "", 0, NULL, NULL, NULL, NULL, NULL, NULL},
 +      /* 9: CD_FACEMAP */
 +      {sizeof(int), "", 0, NULL, NULL, NULL, NULL, NULL, layerDefault_fmap, NULL},
        /* 10: CD_PROP_FLT */
        {sizeof(MFloatProperty), "MFloatProperty", 1, N_("Float"), layerCopy_propFloat, NULL, NULL, NULL},
        /* 11: CD_PROP_INT */
         layerInterp_origspace_face, layerSwap_origspace_face, layerDefault_origspace_face},
        /* 14: CD_ORCO */
        {sizeof(float) * 3, "", 0, NULL, NULL, NULL, NULL, NULL, NULL},
 -      /* 15: CD_MTEXPOLY */
 +      /* 15: CD_MTEXPOLY */  /* DEPRECATED */
        /* note, when we expose the UV Map / TexFace split to the user, change this back to face Texture */
 -      {sizeof(MTexPoly), "MTexPoly", 1, N_("UVMap") /* "Face Texture" */, NULL, NULL, NULL, NULL, NULL,
 -       NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, layerMaxNum_tface},
 +      {sizeof(int), "", 0, NULL, NULL, NULL, NULL, NULL, NULL},
        /* 16: CD_MLOOPUV */
        {sizeof(MLoopUV), "MLoopUV", 1, N_("UVMap"), NULL, NULL, layerInterp_mloopuv, NULL, NULL,
         layerEqual_mloopuv, layerMultiply_mloopuv, layerInitMinMax_mloopuv, 
  
  static const char *LAYERTYPENAMES[CD_NUMTYPES] = {
        /*   0-4 */ "CDMVert", "CDMSticky", "CDMDeformVert", "CDMEdge", "CDMFace",
 -      /*   5-9 */ "CDMTFace", "CDMCol", "CDOrigIndex", "CDNormal", "CDFlags",
 +      /*   5-9 */ "CDMTFace", "CDMCol", "CDOrigIndex", "CDNormal", "CDFaceMap",
        /* 10-14 */ "CDMFloatProperty", "CDMIntProperty", "CDMStringProperty", "CDOrigSpace", "CDOrco",
        /* 15-19 */ "CDMTexPoly", "CDMLoopUV", "CDMloopCol", "CDTangent", "CDMDisps",
        /* 20-24 */ "CDPreviewMCol", "CDIDMCol", "CDTextureMCol", "CDClothOrco", "CDMRecast",
  
  
  const CustomDataMask CD_MASK_BAREMESH =
 -    CD_MASK_MVERT | CD_MASK_MEDGE | CD_MASK_MLOOP | CD_MASK_MPOLY | CD_MASK_BWEIGHT;
 +    CD_MASK_MVERT | CD_MASK_MEDGE | CD_MASK_MLOOP | CD_MASK_MPOLY | CD_MASK_BWEIGHT | CD_MASK_FACEMAP;
  const CustomDataMask CD_MASK_MESH =
      CD_MASK_MVERT | CD_MASK_MEDGE |
      CD_MASK_MDEFORMVERT |
      CD_MASK_PROP_FLT | CD_MASK_PROP_INT | CD_MASK_PROP_STR | CD_MASK_MDISPS |
      CD_MASK_MLOOPUV | CD_MASK_MLOOPCOL | CD_MASK_MPOLY | CD_MASK_MLOOP |
 -    CD_MASK_MTEXPOLY | CD_MASK_RECAST | CD_MASK_PAINT_MASK |
 +    CD_MASK_RECAST | CD_MASK_PAINT_MASK |
      CD_MASK_GRID_PAINT_MASK | CD_MASK_MVERT_SKIN | CD_MASK_FREESTYLE_EDGE | CD_MASK_FREESTYLE_FACE |
 -    CD_MASK_CUSTOMLOOPNORMAL;
 +    CD_MASK_CUSTOMLOOPNORMAL | CD_MASK_FACEMAP;
  const CustomDataMask CD_MASK_EDITMESH =
      CD_MASK_MDEFORMVERT | CD_MASK_MLOOPUV |
 -    CD_MASK_MLOOPCOL | CD_MASK_MTEXPOLY | CD_MASK_SHAPE_KEYINDEX |
 +    CD_MASK_MLOOPCOL | CD_MASK_SHAPE_KEYINDEX |
      CD_MASK_PROP_FLT | CD_MASK_PROP_INT | CD_MASK_PROP_STR |
      CD_MASK_MDISPS | CD_MASK_SHAPEKEY | CD_MASK_RECAST | CD_MASK_PAINT_MASK |
 -    CD_MASK_GRID_PAINT_MASK | CD_MASK_MVERT_SKIN | CD_MASK_CUSTOMLOOPNORMAL;
 +    CD_MASK_GRID_PAINT_MASK | CD_MASK_MVERT_SKIN | CD_MASK_CUSTOMLOOPNORMAL | CD_MASK_FACEMAP;
  const CustomDataMask CD_MASK_DERIVEDMESH =
      CD_MASK_MDEFORMVERT |
      CD_MASK_PROP_FLT | CD_MASK_PROP_INT | CD_MASK_CLOTH_ORCO |
 -    CD_MASK_MLOOPUV | CD_MASK_MLOOPCOL | CD_MASK_MTEXPOLY | CD_MASK_PREVIEW_MLOOPCOL |
 +    CD_MASK_MLOOPUV | CD_MASK_MLOOPCOL | CD_MASK_PREVIEW_MLOOPCOL |
      CD_MASK_PROP_STR | CD_MASK_ORIGSPACE | CD_MASK_ORIGSPACE_MLOOP | CD_MASK_ORCO | CD_MASK_TANGENT |
      CD_MASK_PREVIEW_MCOL | CD_MASK_SHAPEKEY | CD_MASK_RECAST |
      CD_MASK_ORIGINDEX | CD_MASK_MVERT_SKIN | CD_MASK_FREESTYLE_EDGE | CD_MASK_FREESTYLE_FACE |
 -    CD_MASK_CUSTOMLOOPNORMAL;
 +    CD_MASK_CUSTOMLOOPNORMAL | CD_MASK_FACEMAP;
  const CustomDataMask CD_MASK_BMESH =
 -    CD_MASK_MLOOPUV | CD_MASK_MLOOPCOL | CD_MASK_MTEXPOLY |
 +    CD_MASK_MLOOPUV | CD_MASK_MLOOPCOL |
      CD_MASK_MDEFORMVERT | CD_MASK_PROP_FLT | CD_MASK_PROP_INT |
      CD_MASK_PROP_STR | CD_MASK_SHAPEKEY | CD_MASK_SHAPE_KEYINDEX | CD_MASK_MDISPS |
      CD_MASK_CREASE | CD_MASK_BWEIGHT | CD_MASK_RECAST | CD_MASK_PAINT_MASK |
      CD_MASK_GRID_PAINT_MASK | CD_MASK_MVERT_SKIN | CD_MASK_FREESTYLE_EDGE | CD_MASK_FREESTYLE_FACE |
 -    CD_MASK_CUSTOMLOOPNORMAL;
 +      CD_MASK_CUSTOMLOOPNORMAL | CD_MASK_FACEMAP;
  /**
   * cover values copied by #BKE_mesh_loops_to_tessdata
   */
  const CustomDataMask CD_MASK_FACECORNERS =
 -    CD_MASK_MTFACE | CD_MASK_MTEXPOLY | CD_MASK_MLOOPUV |
 +    CD_MASK_MTFACE | CD_MASK_MLOOPUV |
      CD_MASK_MCOL | CD_MASK_MLOOPCOL |
      CD_MASK_PREVIEW_MCOL | CD_MASK_PREVIEW_MLOOPCOL |
      CD_MASK_ORIGSPACE | CD_MASK_ORIGSPACE_MLOOP |
  const CustomDataMask CD_MASK_EVERYTHING =
      CD_MASK_MVERT | CD_MASK_MDEFORMVERT | CD_MASK_MEDGE | CD_MASK_MFACE |
      CD_MASK_MTFACE | CD_MASK_MCOL | CD_MASK_ORIGINDEX | CD_MASK_NORMAL /* | CD_MASK_POLYINDEX */ | CD_MASK_PROP_FLT |
 -    CD_MASK_PROP_INT | CD_MASK_PROP_STR | CD_MASK_ORIGSPACE | CD_MASK_ORCO | CD_MASK_MTEXPOLY | CD_MASK_MLOOPUV |
 +    CD_MASK_PROP_INT | CD_MASK_PROP_STR | CD_MASK_ORIGSPACE | CD_MASK_ORCO | CD_MASK_MLOOPUV |
      CD_MASK_MLOOPCOL | CD_MASK_TANGENT | CD_MASK_MDISPS | CD_MASK_PREVIEW_MCOL | CD_MASK_CLOTH_ORCO | CD_MASK_RECAST |
      /* BMESH ONLY START */
      CD_MASK_MPOLY | CD_MASK_MLOOP | CD_MASK_SHAPE_KEYINDEX | CD_MASK_SHAPEKEY | CD_MASK_BWEIGHT | CD_MASK_CREASE |
      /* BMESH ONLY END */
      CD_MASK_PAINT_MASK | CD_MASK_GRID_PAINT_MASK | CD_MASK_MVERT_SKIN |
      CD_MASK_FREESTYLE_EDGE | CD_MASK_FREESTYLE_FACE |
 -    CD_MASK_MLOOPTANGENT | CD_MASK_TESSLOOPNORMAL | CD_MASK_CUSTOMLOOPNORMAL;
 +    CD_MASK_MLOOPTANGENT | CD_MASK_TESSLOOPNORMAL | CD_MASK_CUSTOMLOOPNORMAL | CD_MASK_FACEMAP;
  
  static const LayerTypeInfo *layerType_getInfo(int type)
  {
@@@ -2492,10 -2502,13 +2492,10 @@@ void CustomData_set(const CustomData *d
  
  /* BMesh functions */
  /* needed to convert to/from different face reps */
 -void CustomData_to_bmeshpoly(CustomData *fdata, CustomData *pdata, CustomData *ldata,
 -                             int totloop, int totpoly)
 +void CustomData_to_bmeshpoly(CustomData *fdata, CustomData *ldata, int totloop)
  {
 -      int i;
 -      for (i = 0; i < fdata->totlayer; i++) {
 +      for (int i = 0; i < fdata->totlayer; i++) {
                if (fdata->layers[i].type == CD_MTFACE) {
 -                      CustomData_add_layer_named(pdata, CD_MTEXPOLY, CD_CALLOC, NULL, totpoly, fdata->layers[i].name);
                        CustomData_add_layer_named(ldata, CD_MLOOPUV, CD_CALLOC, NULL, totloop, fdata->layers[i].name);
                }
                else if (fdata->layers[i].type == CD_MCOL) {
        }
  }
  
 -void CustomData_from_bmeshpoly(CustomData *fdata, CustomData *pdata, CustomData *ldata, int total)
 +void CustomData_from_bmeshpoly(
 +        CustomData *fdata, CustomData *ldata, int total)
  {
        int i;
  
        /* avoid accumulating extra layers */
 -      BLI_assert(!CustomData_from_bmeshpoly_test(fdata, pdata, ldata, false));
 +      BLI_assert(!CustomData_from_bmeshpoly_test(fdata, ldata, false));
  
 -      for (i = 0; i < pdata->totlayer; i++) {
 -              if (pdata->layers[i].type == CD_MTEXPOLY) {
 -                      CustomData_add_layer_named(fdata, CD_MTFACE, CD_CALLOC, NULL, total, pdata->layers[i].name);
 -              }
 -      }
        for (i = 0; i < ldata->totlayer; i++) {
 +              if (ldata->layers[i].type == CD_MLOOPUV) {
 +                      CustomData_add_layer_named(fdata, CD_MTFACE, CD_CALLOC, NULL, total, ldata->layers[i].name);
 +              }
                if (ldata->layers[i].type == CD_MLOOPCOL) {
                        CustomData_add_layer_named(fdata, CD_MCOL, CD_CALLOC, NULL, total, ldata->layers[i].name);
                }
                }
        }
  
 -      CustomData_bmesh_update_active_layers(fdata, pdata, ldata);
 +      CustomData_bmesh_update_active_layers(fdata, ldata);
  }
  
  #ifndef NDEBUG
   * \param fallback: Use when there are no layers to handle,
   * since callers may expect success or failure.
   */
 -bool CustomData_from_bmeshpoly_test(CustomData *fdata, CustomData *pdata, CustomData *ldata, bool fallback)
 +bool CustomData_from_bmeshpoly_test(CustomData *fdata, CustomData *ldata, bool fallback)
  {
        int a_num = 0, b_num = 0;
  #define LAYER_CMP(l_a, t_a, l_b, t_b) \
        ((a_num += CustomData_number_of_layers(l_a, t_a)) == (b_num += CustomData_number_of_layers(l_b, t_b)))
  
 -      if (!LAYER_CMP(pdata, CD_MTEXPOLY, fdata, CD_MTFACE))
 +      if (!LAYER_CMP(ldata, CD_MLOOPUV, fdata, CD_MTFACE))
                return false;
        if (!LAYER_CMP(ldata, CD_MLOOPCOL, fdata, CD_MCOL))
                return false;
  #endif
  
  
 -void CustomData_bmesh_update_active_layers(CustomData *fdata, CustomData *pdata, CustomData *ldata)
 +void CustomData_bmesh_update_active_layers(CustomData *fdata, CustomData *ldata)
  {
        int act;
  
 -      if (CustomData_has_layer(pdata, CD_MTEXPOLY)) {
 -              act = CustomData_get_active_layer(pdata, CD_MTEXPOLY);
 -              CustomData_set_layer_active(ldata, CD_MLOOPUV, act);
 +      if (CustomData_has_layer(ldata, CD_MLOOPUV)) {
 +              act = CustomData_get_active_layer(ldata, CD_MLOOPUV);
                CustomData_set_layer_active(fdata, CD_MTFACE, act);
  
 -              act = CustomData_get_render_layer(pdata, CD_MTEXPOLY);
 -              CustomData_set_layer_render(ldata, CD_MLOOPUV, act);
 +              act = CustomData_get_render_layer(ldata, CD_MLOOPUV);
                CustomData_set_layer_render(fdata, CD_MTFACE, act);
  
 -              act = CustomData_get_clone_layer(pdata, CD_MTEXPOLY);
 -              CustomData_set_layer_clone(ldata, CD_MLOOPUV, act);
 +              act = CustomData_get_clone_layer(ldata, CD_MLOOPUV);
                CustomData_set_layer_clone(fdata, CD_MTFACE, act);
  
 -              act = CustomData_get_stencil_layer(pdata, CD_MTEXPOLY);
 -              CustomData_set_layer_stencil(ldata, CD_MLOOPUV, act);
 +              act = CustomData_get_stencil_layer(ldata, CD_MLOOPUV);
                CustomData_set_layer_stencil(fdata, CD_MTFACE, act);
        }
  
   * used by do_versions in readfile.c when creating pdata and ldata for pre-bmesh
   * meshes and needed to preserve active/render/clone/stencil flags set in pre-bmesh files
   */
 -void CustomData_bmesh_do_versions_update_active_layers(CustomData *fdata, CustomData *pdata, CustomData *ldata)
 +void CustomData_bmesh_do_versions_update_active_layers(CustomData *fdata, CustomData *ldata)
  {
        int act;
  
        if (CustomData_has_layer(fdata, CD_MTFACE)) {
                act = CustomData_get_active_layer(fdata, CD_MTFACE);
 -              CustomData_set_layer_active(pdata, CD_MTEXPOLY, act);
                CustomData_set_layer_active(ldata, CD_MLOOPUV, act);
  
                act = CustomData_get_render_layer(fdata, CD_MTFACE);
 -              CustomData_set_layer_render(pdata, CD_MTEXPOLY, act);
                CustomData_set_layer_render(ldata, CD_MLOOPUV, act);
  
                act = CustomData_get_clone_layer(fdata, CD_MTFACE);
 -              CustomData_set_layer_clone(pdata, CD_MTEXPOLY, act);
                CustomData_set_layer_clone(ldata, CD_MLOOPUV, act);
  
                act = CustomData_get_stencil_layer(fdata, CD_MTFACE);
 -              CustomData_set_layer_stencil(pdata, CD_MTEXPOLY, act);
                CustomData_set_layer_stencil(ldata, CD_MLOOPUV, act);
        }
  
index 1ce5b08c522ea1b87d0c86989890a86509254f97,fd5159bb76c8cdcc96514edb9003ceb20fa0d5c4..10668c9ea2257522d3bed51fef9c31c0eb121dc8
@@@ -56,8 -56,7 +56,8 @@@
  #include "BKE_screen.h"
  #include "BKE_idprop.h"
  
 -#include "BIF_gl.h"
 +#include "GPU_glew.h"
 +#include "GPU_matrix.h"
  
  #include "BLF_api.h"
  #include "BLT_translation.h"
@@@ -500,7 -499,6 +500,7 @@@ static int ui_but_calc_float_precision(
  static void ui_draw_linkline(uiLinkLine *line, int highlightActiveLines, int dashInactiveLines)
  {
        rcti rect;
 +      float color[4] = {1.0f};
  
        if (line->from == NULL || line->to == NULL) return;
        
        rect.ymax = BLI_rctf_cent_y(&line->to->rect);
        
        if (dashInactiveLines)
 -              UI_ThemeColor(TH_GRID);
 +              UI_GetThemeColor4fv(TH_GRID, color);
        else if (line->flag & UI_SELECT)
 -              glColor3ub(100, 100, 100);
 +              rgba_float_args_set_ch(color, 100, 100, 100, 255);
        else if (highlightActiveLines && ((line->from->flag & UI_ACTIVE) || (line->to->flag & UI_ACTIVE)))
 -              UI_ThemeColor(TH_TEXT_HI);
 +              UI_GetThemeColor4fv(TH_TEXT_HI, color);
        else
 -              glColor3ub(0, 0, 0);
 +              rgba_float_args_set_ch(color, 0, 0, 0, 255);
  
 -      ui_draw_link_bezier(&rect);
 +      ui_draw_link_bezier(&rect, color);
  }
  
  static void ui_draw_links(uiBlock *block)
@@@ -1383,9 -1381,11 +1383,9 @@@ void UI_block_draw(const bContext *C, u
        ui_but_to_pixelrect(&rect, ar, block, NULL);
        
        /* pixel space for AA widgets */
 -      glMatrixMode(GL_PROJECTION);
 -      glPushMatrix();
 -      glMatrixMode(GL_MODELVIEW);
 -      glPushMatrix();
 -      glLoadIdentity();
 +      gpuPushProjectionMatrix();
 +      gpuPushMatrix();
 +      gpuLoadIdentity();
  
        wmOrtho2_region_pixelspace(ar);
        
        }
        
        /* restore matrix */
 -      glMatrixMode(GL_PROJECTION);
 -      glPopMatrix();
 -      glMatrixMode(GL_MODELVIEW);
 -      glPopMatrix();
 +      gpuPopProjectionMatrix();
 +      gpuPopMatrix();
  
        if (multisample_enabled)
                glEnable(GL_MULTISAMPLE);
@@@ -1465,7 -1467,6 +1465,7 @@@ int ui_but_is_pushed_ex(uiBut *but, dou
                                break;
                        case UI_BTYPE_ROW:
                        case UI_BTYPE_LISTROW:
 +                      case UI_BTYPE_TAB:
                                UI_GET_BUT_VALUE_INIT(but, *value);
                                /* support for rna enum buts */
                                if (but->rnaprop && (RNA_property_flag(but->rnaprop) & PROP_ENUM_FLAG)) {
@@@ -1936,13 -1937,14 +1936,14 @@@ void ui_but_value_set(uiBut *but, doubl
        else {
                /* first do rounding */
                if (but->pointype == UI_BUT_POIN_CHAR) {
-                       value = (char)floor(value + 0.5);
+                       value = round_db_to_uchar_clamp(value);
                }
                else if (but->pointype == UI_BUT_POIN_SHORT) {
-                       value = (short)floor(value + 0.5);
+                       value = round_db_to_short_clamp(value);
+               }
+               else if (but->pointype == UI_BUT_POIN_INT) {
+                       value = round_db_to_int_clamp(value);
                }
-               else if (but->pointype == UI_BUT_POIN_INT)
-                       value = (int)floor(value + 0.5);
                else if (but->pointype == UI_BUT_POIN_FLOAT) {
                        float fval = (float)value;
                        if (fval >= -0.00001f && fval <= 0.00001f) fval = 0.0f;  /* prevent negative zero */
@@@ -2404,7 -2406,7 +2405,7 @@@ static void ui_but_string_free_internal
  
  bool ui_but_string_set(bContext *C, uiBut *but, const char *str)
  {
 -      if (but->rnaprop && ELEM(but->type, UI_BTYPE_TEXT, UI_BTYPE_SEARCH_MENU)) {
 +      if (but->rnaprop && but->rnapoin.data && ELEM(but->type, UI_BTYPE_TEXT, UI_BTYPE_SEARCH_MENU)) {
                if (RNA_property_editable(&but->rnapoin, but->rnaprop)) {
                        PropertyType type;
  
        }
        else if (but->type == UI_BTYPE_TEXT) {
                /* string */
 -              if (ui_but_is_utf8(but)) BLI_strncpy_utf8(but->poin, str, but->hardmax);
 -              else BLI_strncpy(but->poin, str, but->hardmax);
 +              if (!but->poin || (str[0] == '\0')) {
 +                      str = "";
 +              }
 +              else if (ui_but_is_utf8(but)) {
 +                      BLI_strncpy_utf8(but->poin, str, but->hardmax);
 +              }
 +              else {
 +                      BLI_strncpy(but->poin, str, but->hardmax);
 +              }
  
                return true;
        }
@@@ -2667,10 -2662,6 +2668,10 @@@ static void ui_but_free(const bContext 
                MEM_freeN(but->tip_argN);
        }
  
 +      if (!but->editstr && but->free_search_arg) {
 +              MEM_SAFE_FREE(but->search_arg);
 +      }
 +
        if (but->active) {
                /* XXX solve later, buttons should be free-able without context ideally,
                 * however they may have open tooltips or popup windows, which need to
@@@ -2823,13 -2814,11 +2824,13 @@@ uiBlock *UI_block_begin(const bContext 
                block->aspect = 2.0f / fabsf(getsizex * block->winmat[0][0]);
        }
        else {
 +              const bScreen *screen = WM_window_get_active_screen(window);
 +
                /* no subwindow created yet, for menus for example, so we
                 * use the main window instead, since buttons are created
                 * there anyway */
 -              wm_subwindow_matrix_get(window, window->screen->mainwin, block->winmat);
 -              wm_subwindow_size_get(window, window->screen->mainwin, &getsizex, &getsizey);
 +              wm_subwindow_matrix_get(window, screen->mainwin, block->winmat);
 +              wm_subwindow_size_get(window, screen->mainwin, &getsizex, &getsizey);
  
                block->aspect = 2.0f / fabsf(getsizex * block->winmat[0][0]);
                block->auto_open = true;
@@@ -3547,7 -3536,7 +3548,7 @@@ static uiBut *ui_def_but_rna
        }
  
        const char *info;
 -      if (!RNA_property_editable_info(&but->rnapoin, prop, &info)) {
 +      if (but->rnapoin.data && !RNA_property_editable_info(&but->rnapoin, prop, &info)) {
                ui_def_but_rna__disable(but, info);
        }
  
@@@ -4446,7 -4435,7 +4447,7 @@@ static void operator_enum_search_cb(con
                for (item = item_array; item->identifier; item++) {
                        /* note: need to give the index rather than the identifier because the enum can be freed */
                        if (BLI_strcasestr(item->name, str)) {
 -                              if (false == UI_search_item_add(items, item->name, SET_INT_IN_POINTER(item->value), 0))
 +                              if (false == UI_search_item_add(items, item->name, SET_INT_IN_POINTER(item->value), item->icon))
                                        break;
                        }
                }
@@@ -4570,7 -4559,7 +4571,7 @@@ void UI_but_string_info_get(bContext *C
                                tmp = BLI_strdup(RNA_property_identifier(but->rnaprop));
                }
                else if (type == BUT_GET_RNASTRUCT_IDENTIFIER) {
 -                      if (but->rnaprop)
 +                      if (but->rnaprop && but->rnapoin.data)
                                tmp = BLI_strdup(RNA_struct_identifier(but->rnapoin.type));
                        else if (but->optype)
                                tmp = BLI_strdup(but->optype->idname);
                        if (ptr && prop) {
                                if (!item) {
                                        int i;
 -                                      
 +
 +                                      /* so the context is passed to itemf functions */
 +                                      WM_operator_properties_sanitize(ptr, false);
 +
                                        RNA_property_enum_items_gettexted(C, ptr, prop, &items, &totitems, &free_items);
                                        for (i = 0, item = items; i < totitems; i++, item++) {
                                                if (item->identifier[0] && item->value == value)
index e94b724043fe03656a3593d51ecef27ac169fa88,3d6dd9a5f35efdcedda1a488a21de765daba90a4..524e9935760aa7915526aac0cd142f38249c0468
@@@ -68,12 -68,11 +68,12 @@@ EnumPropertyItem rna_enum_mesh_delimit_
  #include "BLI_math.h"
  
  #include "BKE_customdata.h"
 -#include "BKE_depsgraph.h"
  #include "BKE_main.h"
  #include "BKE_mesh.h"
  #include "BKE_report.h"
  
 +#include "DEG_depsgraph.h"
 +
  #include "ED_mesh.h" /* XXX Bad level call */
  
  #include "WM_api.h"
@@@ -187,10 -186,12 +187,10 @@@ static void rna_MeshEdgeLayer_name_set(
        rna_cd_layer_name_set(rna_mesh_edata(ptr), (CustomDataLayer *)ptr->data, value);
  }
  #endif
 -#if 0
  static void rna_MeshPolyLayer_name_set(PointerRNA *ptr, const char *value)
  {
        rna_cd_layer_name_set(rna_mesh_pdata(ptr), (CustomDataLayer *)ptr->data, value);
  }
 -#endif
  static void rna_MeshLoopLayer_name_set(PointerRNA *ptr, const char *value)
  {
        rna_cd_layer_name_set(rna_mesh_ldata(ptr), (CustomDataLayer *)ptr->data, value);
@@@ -223,7 -224,7 +223,7 @@@ static void rna_Mesh_update_data(Main *
  
        /* cheating way for importers to avoid slow updates */
        if (id->us > 0) {
 -              DAG_id_tag_update(id, 0);
 +              DEG_id_tag_update(id, 0);
                WM_main_add_notifier(NC_GEOM | ND_DATA, id);
        }
  }
@@@ -237,20 -238,6 +237,20 @@@ static void rna_Mesh_update_data_edit_c
        }
  }
  
 +static void rna_Mesh_update_data_edit_weight(Main *bmain, Scene *scene, PointerRNA *ptr)
 +{
 +      BKE_mesh_batch_cache_dirty(rna_mesh(ptr), BKE_MESH_BATCH_DIRTY_ALL);
 +
 +      rna_Mesh_update_data(bmain, scene, ptr);
 +}
 +
 +
 +static void rna_Mesh_update_data_edit_active_color(Main *bmain, Scene *scene, PointerRNA *ptr)
 +{
 +      BKE_mesh_batch_cache_dirty(rna_mesh(ptr), BKE_MESH_BATCH_DIRTY_ALL);
 +
 +      rna_Mesh_update_data(bmain, scene, ptr);
 +}
  static void rna_Mesh_update_select(Main *UNUSED(bmain), Scene *UNUSED(scene), PointerRNA *ptr)
  {
        ID *id = ptr->id.data;
@@@ -276,9 -263,6 +276,9 @@@ static void rna_Mesh_update_vertmask(Ma
        if ((me->editflag & ME_EDIT_PAINT_VERT_SEL) && (me->editflag & ME_EDIT_PAINT_FACE_SEL)) {
                me->editflag &= ~ME_EDIT_PAINT_FACE_SEL;
        }
 +
 +      BKE_mesh_batch_cache_dirty(me, BKE_MESH_BATCH_DIRTY_ALL);
 +
        rna_Mesh_update_draw(bmain, scene, ptr);
  }
  
@@@ -288,9 -272,6 +288,9 @@@ static void rna_Mesh_update_facemask(Ma
        if ((me->editflag & ME_EDIT_PAINT_VERT_SEL) && (me->editflag & ME_EDIT_PAINT_FACE_SEL)) {
                me->editflag &= ~ME_EDIT_PAINT_VERT_SEL;
        }
 +
 +      BKE_mesh_batch_cache_dirty(me, BKE_MESH_BATCH_DIRTY_ALL);
 +
        rna_Mesh_update_draw(bmain, scene, ptr);
  }
  
@@@ -323,7 -304,7 +323,7 @@@ static float rna_MeshVertex_bevel_weigh
  static void rna_MeshVertex_bevel_weight_set(PointerRNA *ptr, float value)
  {
        MVert *mvert = (MVert *)ptr->data;
-       mvert->bweight = (char)(CLAMPIS(value * 255.0f, 0, 255));
+       mvert->bweight = round_fl_to_uchar_clamp(value * 255.0f);
  }
  
  static float rna_MEdge_bevel_weight_get(PointerRNA *ptr)
  static void rna_MEdge_bevel_weight_set(PointerRNA *ptr, float value)
  {
        MEdge *medge = (MEdge *)ptr->data;
-       medge->bweight = (char)(CLAMPIS(value * 255.0f, 0, 255));
+       medge->bweight = round_fl_to_uchar_clamp(value * 255.0f);
  }
  
  static float rna_MEdge_crease_get(PointerRNA *ptr)
  static void rna_MEdge_crease_set(PointerRNA *ptr, float value)
  {
        MEdge *medge = (MEdge *)ptr->data;
-       medge->crease = (char)(CLAMPIS(value * 255.0f, 0, 255));
+       medge->crease = round_fl_to_uchar_clamp(value * 255.0f);
  }
  
  static void rna_MeshLoop_normal_get(PointerRNA *ptr, float *values)
@@@ -605,9 -586,9 +605,9 @@@ static void rna_MeshColor_color1_set(Po
  {
        MCol *mcol = (MCol *)ptr->data;
  
-       (&mcol[0].r)[2] = (char)(CLAMPIS(values[0] * 255.0f, 0, 255));
-       (&mcol[0].r)[1] = (char)(CLAMPIS(values[1] * 255.0f, 0, 255));
-       (&mcol[0].r)[0] = (char)(CLAMPIS(values[2] * 255.0f, 0, 255));
+       (&mcol[0].r)[2] = round_fl_to_uchar_clamp(values[0] * 255.0f);
+       (&mcol[0].r)[1] = round_fl_to_uchar_clamp(values[1] * 255.0f);
+       (&mcol[0].r)[0] = round_fl_to_uchar_clamp(values[2] * 255.0f);
  }
  
  static void rna_MeshColor_color2_get(PointerRNA *ptr, float *values)
@@@ -623,9 -604,9 +623,9 @@@ static void rna_MeshColor_color2_set(Po
  {
        MCol *mcol = (MCol *)ptr->data;
  
-       (&mcol[1].r)[2] = (char)(CLAMPIS(values[0] * 255.0f, 0, 255));
-       (&mcol[1].r)[1] = (char)(CLAMPIS(values[1] * 255.0f, 0, 255));
-       (&mcol[1].r)[0] = (char)(CLAMPIS(values[2] * 255.0f, 0, 255));
+       (&mcol[1].r)[2] = round_fl_to_uchar_clamp(values[0] * 255.0f);
+       (&mcol[1].r)[1] = round_fl_to_uchar_clamp(values[1] * 255.0f);
+       (&mcol[1].r)[0] = round_fl_to_uchar_clamp(values[2] * 255.0f);
  }
  
  static void rna_MeshColor_color3_get(PointerRNA *ptr, float *values)
@@@ -641,9 -622,9 +641,9 @@@ static void rna_MeshColor_color3_set(Po
  {
        MCol *mcol = (MCol *)ptr->data;
  
-       (&mcol[2].r)[2] = (char)(CLAMPIS(values[0] * 255.0f, 0, 255));
-       (&mcol[2].r)[1] = (char)(CLAMPIS(values[1] * 255.0f, 0, 255));
-       (&mcol[2].r)[0] = (char)(CLAMPIS(values[2] * 255.0f, 0, 255));
+       (&mcol[2].r)[2] = round_fl_to_uchar_clamp(values[0] * 255.0f);
+       (&mcol[2].r)[1] = round_fl_to_uchar_clamp(values[1] * 255.0f);
+       (&mcol[2].r)[0] = round_fl_to_uchar_clamp(values[2] * 255.0f);
  }
  
  static void rna_MeshColor_color4_get(PointerRNA *ptr, float *values)
@@@ -659,9 -640,9 +659,9 @@@ static void rna_MeshColor_color4_set(Po
  {
        MCol *mcol = (MCol *)ptr->data;
  
-       (&mcol[3].r)[2] = (char)(CLAMPIS(values[0] * 255.0f, 0, 255));
-       (&mcol[3].r)[1] = (char)(CLAMPIS(values[1] * 255.0f, 0, 255));
-       (&mcol[3].r)[0] = (char)(CLAMPIS(values[2] * 255.0f, 0, 255));
+       (&mcol[3].r)[2] = round_fl_to_uchar_clamp(values[0] * 255.0f);
+       (&mcol[3].r)[1] = round_fl_to_uchar_clamp(values[1] * 255.0f);
+       (&mcol[3].r)[0] = round_fl_to_uchar_clamp(values[2] * 255.0f);
  }
  
  static void rna_MeshLoopColor_color_get(PointerRNA *ptr, float *values)
@@@ -677,9 -658,9 +677,9 @@@ static void rna_MeshLoopColor_color_set
  {
        MLoopCol *mcol = (MLoopCol *)ptr->data;
  
-       (&mcol->r)[0] = (char)(CLAMPIS(values[0] * 255.0f, 0, 255));
-       (&mcol->r)[1] = (char)(CLAMPIS(values[1] * 255.0f, 0, 255));
-       (&mcol->r)[2] = (char)(CLAMPIS(values[2] * 255.0f, 0, 255));
+       (&mcol->r)[0] = round_fl_to_uchar_clamp(values[0] * 255.0f);
+       (&mcol->r)[1] = round_fl_to_uchar_clamp(values[1] * 255.0f);
+       (&mcol->r)[2] = round_fl_to_uchar_clamp(values[2] * 255.0f);
  }
  
  static int rna_Mesh_texspace_editable(PointerRNA *ptr, const char **UNUSED(r_info))
@@@ -767,6 -748,13 +767,6 @@@ static void rna_CustomDataLayer_active_
        if (render) CustomData_set_layer_render(data, type, n);
        else CustomData_set_layer_active(data, type, n);
  
 -      /* sync loop layer */
 -      if (type == CD_MTEXPOLY) {
 -              CustomData *ldata = rna_mesh_ldata(ptr);
 -              if (render) CustomData_set_layer_render(ldata, CD_MLOOPUV, n);
 -              else CustomData_set_layer_active(ldata, CD_MLOOPUV, n);
 -      }
 -
        BKE_mesh_update_customdata_pointers(me, true);
  }
  
@@@ -871,36 -859,6 +871,36 @@@ static int rna_MeshUVLoopLayer_data_len
        return (me->edit_btmesh) ? 0 : me->totloop;
  }
  
 +static int rna_MeshUVLoopLayer_active_render_get(PointerRNA *ptr)
 +{
 +      return rna_CustomDataLayer_active_get(ptr, rna_mesh_ldata(ptr), CD_MLOOPUV, 1);
 +}
 +
 +static int rna_MeshUVLoopLayer_active_get(PointerRNA *ptr)
 +{
 +      return rna_CustomDataLayer_active_get(ptr, rna_mesh_ldata(ptr), CD_MLOOPUV, 0);
 +}
 +
 +static int rna_MeshUVLoopLayer_clone_get(PointerRNA *ptr)
 +{
 +      return rna_CustomDataLayer_clone_get(ptr, rna_mesh_ldata(ptr), CD_MLOOPUV);
 +}
 +
 +static void rna_MeshUVLoopLayer_active_render_set(PointerRNA *ptr, int value)
 +{
 +      rna_CustomDataLayer_active_set(ptr, rna_mesh_ldata(ptr), value, CD_MLOOPUV, 1);
 +}
 +
 +static void rna_MeshUVLoopLayer_active_set(PointerRNA *ptr, int value)
 +{
 +      rna_CustomDataLayer_active_set(ptr, rna_mesh_ldata(ptr), value, CD_MLOOPUV, 0);
 +}
 +
 +static void rna_MeshUVLoopLayer_clone_set(PointerRNA *ptr, int value)
 +{
 +      rna_CustomDataLayer_clone_set(ptr, rna_mesh_ldata(ptr), value, CD_MLOOPUV);
 +}
 +
  /* face uv_textures */
  
  DEFINE_CUSTOMDATA_LAYER_COLLECTION(tessface_uv_texture, fdata, CD_MTFACE)
@@@ -952,6 -910,57 +952,6 @@@ static void rna_MeshTextureFaceLayer_cl
        rna_CustomDataLayer_clone_set(ptr, rna_mesh_fdata(ptr), value, CD_MTFACE);
  }
  
 -/* poly uv_textures */
 -
 -DEFINE_CUSTOMDATA_LAYER_COLLECTION(uv_texture, pdata, CD_MTEXPOLY)
 -DEFINE_CUSTOMDATA_LAYER_COLLECTION_ACTIVEITEM(uv_texture, pdata, CD_MTEXPOLY, active, MeshTexturePolyLayer)
 -DEFINE_CUSTOMDATA_LAYER_COLLECTION_ACTIVEITEM(uv_texture, pdata, CD_MTEXPOLY, clone, MeshTexturePolyLayer)
 -DEFINE_CUSTOMDATA_LAYER_COLLECTION_ACTIVEITEM(uv_texture, pdata, CD_MTEXPOLY, stencil, MeshTexturePolyLayer)
 -DEFINE_CUSTOMDATA_LAYER_COLLECTION_ACTIVEITEM(uv_texture, pdata, CD_MTEXPOLY, render, MeshTexturePolyLayer)
 -
 -static void rna_MeshTexturePolyLayer_data_begin(CollectionPropertyIterator *iter, PointerRNA *ptr)
 -{
 -      Mesh *me = rna_mesh(ptr);
 -      CustomDataLayer *layer = (CustomDataLayer *)ptr->data;
 -      rna_iterator_array_begin(iter, layer->data, sizeof(MTexPoly), (me->edit_btmesh) ? 0 : me->totpoly, 0, NULL);
 -}
 -
 -static int rna_MeshTexturePolyLayer_data_length(PointerRNA *ptr)
 -{
 -      Mesh *me = rna_mesh(ptr);
 -      return (me->edit_btmesh) ? 0 : me->totpoly;
 -}
 -
 -static int rna_MeshTexturePolyLayer_active_render_get(PointerRNA *ptr)
 -{
 -      return rna_CustomDataLayer_active_get(ptr, rna_mesh_pdata(ptr), CD_MTEXPOLY, 1);
 -}
 -
 -static int rna_MeshTexturePolyLayer_active_get(PointerRNA *ptr)
 -{
 -      return rna_CustomDataLayer_active_get(ptr, rna_mesh_pdata(ptr), CD_MTEXPOLY, 0);
 -}
 -
 -static int rna_MeshTexturePolyLayer_clone_get(PointerRNA *ptr)
 -{
 -      return rna_CustomDataLayer_clone_get(ptr, rna_mesh_pdata(ptr), CD_MTEXPOLY);
 -}
 -
 -static void rna_MeshTexturePolyLayer_active_render_set(PointerRNA *ptr, int value)
 -{
 -      rna_CustomDataLayer_active_set(ptr, rna_mesh_pdata(ptr), value, CD_MTEXPOLY, 1);
 -}
 -
 -static void rna_MeshTexturePolyLayer_active_set(PointerRNA *ptr, int value)
 -{
 -      rna_CustomDataLayer_active_set(ptr, rna_mesh_pdata(ptr), value, CD_MTEXPOLY, 0);
 -}
 -
 -static void rna_MeshTexturePolyLayer_clone_set(PointerRNA *ptr, int value)
 -{
 -      rna_CustomDataLayer_clone_set(ptr, rna_mesh_pdata(ptr), value, CD_MTEXPOLY);
 -}
 -
  /* vertex_color_layers */
  
  DEFINE_CUSTOMDATA_LAYER_COLLECTION(tessface_vertex_color, fdata, CD_MCOL)
@@@ -1175,75 -1184,42 +1175,75 @@@ static int rna_MeshPaintMaskLayer_data_
  
  /* End paint mask */
  
 -static void rna_TexturePoly_image_set(PointerRNA *ptr, PointerRNA value)
 +/* Face maps */
 +
 +DEFINE_CUSTOMDATA_LAYER_COLLECTION(face_map, pdata, CD_FACEMAP)
 +DEFINE_CUSTOMDATA_LAYER_COLLECTION_ACTIVEITEM(face_map, pdata, CD_FACEMAP, active, MeshFaceMapLayer)
 +
 +static char *rna_MeshFaceMapLayer_path(PointerRNA *ptr)
  {
 -      MTexPoly *tf = (MTexPoly *)ptr->data;
 -      ID *id = value.data;
 +      CustomDataLayer *cdl = ptr->data;
 +      char name_esc[sizeof(cdl->name) * 2];
 +      BLI_strescape(name_esc, cdl->name, sizeof(name_esc));
 +      return BLI_sprintfN("face_maps[\"%s\"]", name_esc);
 +}
 +
 +static void rna_MeshFaceMapLayer_data_begin(CollectionPropertyIterator *iter, PointerRNA *ptr)
 +{
 +      Mesh *me = rna_mesh(ptr);
 +      CustomDataLayer *layer = (CustomDataLayer *)ptr->data;
 +      rna_iterator_array_begin(iter, layer->data, sizeof(int), me->totpoly, 0, NULL);
 +}
 +
 +static int rna_MeshFaceMapLayer_data_length(PointerRNA *ptr)
 +{
 +      Mesh *me = rna_mesh(ptr);
 +      return me->totpoly;
 +}
  
 -      if (id) {
 -              /* special exception here, individual faces don't count
 -               * as reference, but we do ensure the refcount is not zero */
 -              if (id->us == 0)
 -                      id_us_plus(id);
 -              else
 -                      id_lib_extern(id);
 +static PointerRNA rna_Mesh_face_map_new(struct Mesh *me, ReportList *reports, const char *name)
 +{
 +      if (BKE_mesh_ensure_facemap_customdata(me) == false) {
 +              BKE_report(reports, RPT_ERROR, "Currently only single face-map layers are supported");
 +              return PointerRNA_NULL;
        }
  
 -      tf->tpage = (struct Image *)id;
 +      CustomData *pdata = rna_mesh_pdata_helper(me);
 +
 +      int index = CustomData_get_layer_index(pdata, CD_FACEMAP);
 +      BLI_assert(index != -1);
 +      CustomDataLayer *cdl = &pdata->layers[index];
 +      rna_cd_layer_name_set(pdata, cdl, name);
 +
 +      PointerRNA ptr;
 +      RNA_pointer_create(&me->id, &RNA_MeshFaceMapLayer, cdl, &ptr);
 +      return ptr;
  }
  
 -/* while this is supposed to be readonly,
 - * keep it to support importers that only make tessfaces */
 -static void rna_TextureFace_image_set(PointerRNA *ptr, PointerRNA value)
 -{
 -      MTFace *tf = (MTFace *)ptr->data;
 -      ID *id = value.data;
 -
 -      if (id) {
 -              /* special exception here, individual faces don't count
 -               * as reference, but we do ensure the refcount is not zero */
 -              if (id->us == 0)
 -                      id_us_plus(id);
 -              else
 -                      id_lib_extern(id);
 +static void rna_Mesh_face_map_remove(struct Mesh *me, ReportList *reports, struct CustomDataLayer *layer)
 +{
 +      /* just for sanity check */
 +      {
 +              CustomData *pdata = rna_mesh_pdata_helper(me);
 +              int index = CustomData_get_layer_index(pdata, CD_FACEMAP);
 +              if (index != -1) {
 +                      CustomDataLayer *layer_test = &pdata->layers[index];
 +                      if (layer != layer_test) {
 +                              /* don't show name, its likely freed memory */
 +                              BKE_report(reports, RPT_ERROR, "FaceMap not in mesh");
 +                              return;
 +                      }
 +              }
        }
  
 -      tf->tpage = (struct Image *)id;
 +      if (BKE_mesh_clear_facemap_customdata(me) == false) {
 +              BKE_reportf(reports, RPT_ERROR, "Error removing face-map");
 +      }
  }
  
 +/* End face maps */
 +
 +
  static int rna_MeshTessFace_verts_get_length(PointerRNA *ptr, int length[RNA_MAX_ARRAY_DIMENSION])
  {
        MFace *face = (MFace *)ptr->data;
@@@ -1394,6 -1370,14 +1394,6 @@@ static char *rna_MeshTextureFaceLayer_p
        return BLI_sprintfN("tessface_uv_textures[\"%s\"]", name_esc);
  }
  
 -static char *rna_MeshTexturePolyLayer_path(PointerRNA *ptr)
 -{
 -      CustomDataLayer *cdl = ptr->data;
 -      char name_esc[sizeof(cdl->name) * 2];
 -      BLI_strescape(name_esc, cdl->name, sizeof(name_esc));
 -      return BLI_sprintfN("uv_textures[\"%s\"]", name_esc);
 -}
 -
  static char *rna_VertCustomData_data_path(PointerRNA *ptr, const char *collection, int type)
  {
        CustomDataLayer *cdl;
@@@ -1489,6 -1473,11 +1489,6 @@@ static char *rna_MeshTextureFace_path(P
        return rna_FaceCustomData_data_path(ptr, "tessface_uv_textures", CD_MTFACE);
  }
  
 -static char *rna_MeshTexturePoly_path(PointerRNA *ptr)
 -{
 -      return rna_PolyCustomData_data_path(ptr, "uv_textures", CD_MTEXPOLY);
 -}
 -
  static char *rna_MeshColorLayer_path(PointerRNA *ptr)
  {
        CustomDataLayer *cdl = ptr->data;
@@@ -1676,12 -1665,6 +1676,12 @@@ void rna_MeshStringProperty_s_set(Point
        MStringProperty *ms = (MStringProperty *)ptr->data;
        BLI_strncpy(ms->s, value, sizeof(ms->s));
  }
 +
 +static char *rna_MeshFaceMap_path(PointerRNA *ptr)
 +{
 +      return rna_PolyCustomData_data_path(ptr, "face_maps", CD_FACEMAP);
 +}
 +
  /***************************************/
  
  static int rna_Mesh_tot_vert_get(PointerRNA *ptr)
@@@ -1775,23 -1758,23 +1775,23 @@@ DEFINE_CUSTOMDATA_PROPERTY_API(polygon
  DEFINE_CUSTOMDATA_PROPERTY_API(polygon, string, CD_PROP_STR, pdata, totpoly, MeshPolygonStringPropertyLayer)
  #undef DEFINE_CUSTOMDATA_PROPERTY_API
  
 -static PointerRNA rna_Mesh_uv_texture_new(struct Mesh *me, const char *name)
 +static PointerRNA rna_Mesh_uv_layers_new(struct Mesh *me, const char *name)
  {
        PointerRNA ptr;
 -      CustomData *pdata;
 +      CustomData *ldata;
        CustomDataLayer *cdl = NULL;
        int index = ED_mesh_uv_texture_add(me, name, false);
  
        if (index != -1) {
 -              pdata = rna_mesh_pdata_helper(me);
 -              cdl = &pdata->layers[CustomData_get_layer_index_n(pdata, CD_MTEXPOLY, index)];
 +              ldata = rna_mesh_ldata_helper(me);
 +              cdl = &ldata->layers[CustomData_get_layer_index_n(ldata, CD_MLOOPUV, index)];
        }
  
 -      RNA_pointer_create(&me->id, &RNA_MeshTexturePolyLayer, cdl, &ptr);
 +      RNA_pointer_create(&me->id, &RNA_MeshUVLoopLayer, cdl, &ptr);
        return ptr;
  }
  
 -static void rna_Mesh_uv_texture_layers_remove(struct Mesh *me, ReportList *reports, CustomDataLayer *layer)
 +static void rna_Mesh_uv_layers_remove(struct Mesh *me, ReportList *reports, CustomDataLayer *layer)
  {
        if (ED_mesh_uv_texture_remove_named(me, layer->name) == false) {
                BKE_reportf(reports, RPT_ERROR, "Texture layer '%s' not found", layer->name);
@@@ -1866,14 -1849,14 +1866,14 @@@ static void UNUSED_FUNCTION(rna_mesh_un
        (void)rna_Mesh_uv_layer_render_index_get;
        (void)rna_Mesh_uv_layer_render_index_set;
        (void)rna_Mesh_uv_layer_render_set;
 -      (void)rna_Mesh_uv_texture_render_get;
 -      (void)rna_Mesh_uv_texture_render_index_get;
 -      (void)rna_Mesh_uv_texture_render_index_set;
 -      (void)rna_Mesh_uv_texture_render_set;
        (void)rna_Mesh_vertex_color_render_get;
        (void)rna_Mesh_vertex_color_render_index_get;
        (void)rna_Mesh_vertex_color_render_index_set;
        (void)rna_Mesh_vertex_color_render_set;
 +      (void)rna_Mesh_face_map_index_range;
 +      (void)rna_Mesh_face_map_active_index_set;
 +      (void)rna_Mesh_face_map_active_index_get;
 +      (void)rna_Mesh_face_map_active_set;
        /* end unused function block */
  }
  
@@@ -1901,7 -1884,7 +1901,7 @@@ static void rna_def_mvert_group(Blender
        prop = RNA_def_property(srna, "weight", PROP_FLOAT, PROP_NONE);
        RNA_def_property_range(prop, 0.0f, 1.0f);
        RNA_def_property_ui_text(prop, "Weight", "Vertex Weight");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 +      RNA_def_property_update(prop, 0, "rna_Mesh_update_data_edit_weight");
  }
  
  static void rna_def_mvert(BlenderRNA *brna)
@@@ -2271,24 -2254,6 +2271,24 @@@ static void rna_def_mloopuv(BlenderRNA 
        RNA_def_property_ui_text(prop, "Name", "Name of UV map");
        RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
  
 +      prop = RNA_def_property(srna, "active", PROP_BOOLEAN, PROP_NONE);
 +      RNA_def_property_boolean_funcs(prop, "rna_MeshUVLoopLayer_active_get", "rna_MeshUVLoopLayer_active_set");
 +      RNA_def_property_ui_text(prop, "Active", "Set the map as active for display and editing");
 +      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 +
 +      prop = RNA_def_property(srna, "active_render", PROP_BOOLEAN, PROP_NONE);
 +      RNA_def_property_boolean_sdna(prop, NULL, "active_rnd", 0);
 +      RNA_def_property_boolean_funcs(prop, "rna_MeshUVLoopLayer_active_render_get",
 +                                     "rna_MeshUVLoopLayer_active_render_set");
 +      RNA_def_property_ui_text(prop, "Active Render", "Set the map as active for rendering");
 +      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 +
 +      prop = RNA_def_property(srna, "active_clone", PROP_BOOLEAN, PROP_NONE);
 +      RNA_def_property_boolean_sdna(prop, NULL, "active_clone", 0);
 +      RNA_def_property_boolean_funcs(prop, "rna_MeshUVLoopLayer_clone_get", "rna_MeshUVLoopLayer_clone_set");
 +      RNA_def_property_ui_text(prop, "Active Clone", "Set the map as active for cloning");
 +      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 +
        srna = RNA_def_struct(brna, "MeshUVLoop", NULL);
        RNA_def_struct_sdna(srna, "MLoopUV");
        RNA_def_struct_path_func(srna, "rna_MeshUVLoop_path");
@@@ -2358,6 -2323,13 +2358,6 @@@ static void rna_def_mtface(BlenderRNA *
        RNA_def_struct_path_func(srna, "rna_MeshTextureFace_path");
        RNA_def_struct_ui_icon(srna, ICON_FACESEL_HLT);
  
 -      prop = RNA_def_property(srna, "image", PROP_POINTER, PROP_NONE);
 -      RNA_def_property_pointer_sdna(prop, NULL, "tpage");
 -      RNA_def_property_pointer_funcs(prop, NULL, "rna_TextureFace_image_set", NULL, NULL);
 -      RNA_def_property_flag(prop, PROP_EDITABLE);
 -      RNA_def_property_ui_text(prop, "Image", "");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 -
        /* these are for editing only, access at loops now */
  #if 0
        prop = RNA_def_property(srna, "select_uv", PROP_BOOLEAN, PROP_NONE);
  
  }
  
 -static void rna_def_mtexpoly(BlenderRNA *brna)
 -{
 -      StructRNA *srna;
 -      PropertyRNA *prop;
 -#if 0  /* BMESH_TODO: needed later when do another todo */
 -      int uv_dim[] = {4, 2};
 -#endif
 -
 -      srna = RNA_def_struct(brna, "MeshTexturePolyLayer", NULL);
 -      RNA_def_struct_ui_text(srna, "Mesh UV Map", "UV map with assigned image textures in a Mesh data-block");
 -      RNA_def_struct_sdna(srna, "CustomDataLayer");
 -      RNA_def_struct_path_func(srna, "rna_MeshTexturePolyLayer_path");
 -      RNA_def_struct_ui_icon(srna, ICON_GROUP_UVS);
 -
 -      prop = RNA_def_property(srna, "name", PROP_STRING, PROP_NONE);
 -      RNA_def_struct_name_property(srna, prop);
 -      RNA_def_property_string_funcs(prop, NULL, NULL, "rna_MeshUVLayer_name_set");
 -      RNA_def_property_ui_text(prop, "Name", "Name of UV map");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 -
 -      prop = RNA_def_property(srna, "active", PROP_BOOLEAN, PROP_NONE);
 -      RNA_def_property_boolean_funcs(prop, "rna_MeshTexturePolyLayer_active_get", "rna_MeshTexturePolyLayer_active_set");
 -      RNA_def_property_ui_text(prop, "Active", "Set the map as active for display and editing");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 -
 -      prop = RNA_def_property(srna, "active_render", PROP_BOOLEAN, PROP_NONE);
 -      RNA_def_property_boolean_sdna(prop, NULL, "active_rnd", 0);
 -      RNA_def_property_boolean_funcs(prop, "rna_MeshTexturePolyLayer_active_render_get",
 -                                     "rna_MeshTexturePolyLayer_active_render_set");
 -      RNA_def_property_ui_text(prop, "Active Render", "Set the map as active for rendering");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 -
 -      prop = RNA_def_property(srna, "active_clone", PROP_BOOLEAN, PROP_NONE);
 -      RNA_def_property_boolean_sdna(prop, NULL, "active_clone", 0);
 -      RNA_def_property_boolean_funcs(prop, "rna_MeshTexturePolyLayer_clone_get", "rna_MeshTexturePolyLayer_clone_set");
 -      RNA_def_property_ui_text(prop, "Active Clone", "Set the map as active for cloning");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 -
 -      prop = RNA_def_property(srna, "data", PROP_COLLECTION, PROP_NONE);
 -      RNA_def_property_struct_type(prop, "MeshTexturePoly");
 -      RNA_def_property_ui_text(prop, "Data", "");
 -      RNA_def_property_collection_funcs(prop, "rna_MeshTexturePolyLayer_data_begin", "rna_iterator_array_next",
 -                                        "rna_iterator_array_end", "rna_iterator_array_get",
 -                                        "rna_MeshTexturePolyLayer_data_length", NULL, NULL, NULL);
 -
 -      srna = RNA_def_struct(brna, "MeshTexturePoly", NULL);
 -      RNA_def_struct_sdna(srna, "MTexPoly");
 -      RNA_def_struct_ui_text(srna, "Mesh UV Map Face", "UV map and image texture for a face");
 -      RNA_def_struct_path_func(srna, "rna_MeshTexturePoly_path");
 -      RNA_def_struct_ui_icon(srna, ICON_FACESEL_HLT);
 -
 -      prop = RNA_def_property(srna, "image", PROP_POINTER, PROP_NONE);
 -      RNA_def_property_pointer_sdna(prop, NULL, "tpage");
 -      RNA_def_property_pointer_funcs(prop, NULL, "rna_TexturePoly_image_set", NULL, NULL);
 -      RNA_def_property_flag(prop, PROP_EDITABLE);
 -      RNA_def_property_ui_text(prop, "Image", "");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 -
 -#if 0 /* moved to MeshUVLoopLayer */
 -      prop = RNA_def_property(srna, "select_uv", PROP_BOOLEAN, PROP_NONE);
 -      RNA_def_property_boolean_sdna(prop, NULL, "flag", TF_SEL1);
 -      RNA_def_property_array(prop, 4);
 -      RNA_def_property_ui_text(prop, "UV Selected", "");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_select");
 -
 -      prop = RNA_def_property(srna, "pin_uv", PROP_BOOLEAN, PROP_NONE);
 -      RNA_def_property_boolean_sdna(prop, NULL, "unwrap", TF_PIN1);
 -      RNA_def_property_array(prop, 4);
 -      RNA_def_property_ui_text(prop, "UV Pinned", "");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_select");
 -
 -      prop = RNA_def_property(srna, "uv_raw", PROP_FLOAT, PROP_NONE);
 -      RNA_def_property_multi_array(prop, 2, uv_dim);
 -      RNA_def_property_float_sdna(prop, NULL, "uv");
 -      RNA_def_property_ui_text(prop, "UV", "Fixed size UV coordinates array");
 -#endif
 -}
 -
  static void rna_def_mcol(BlenderRNA *brna)
  {
        StructRNA *srna;
@@@ -2878,13 -2928,13 +2878,13 @@@ static void rna_def_loop_colors(Blender
                                       "rna_Mesh_vertex_color_active_set", NULL, NULL);
        RNA_def_property_flag(prop, PROP_EDITABLE | PROP_NEVER_UNLINK);
        RNA_def_property_ui_text(prop, "Active Vertex Color Layer", "Active vertex color layer");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 +      RNA_def_property_update(prop, 0, "rna_Mesh_update_data_edit_active_color");
  
        prop = RNA_def_property(srna, "active_index", PROP_INT, PROP_UNSIGNED);
        RNA_def_property_int_funcs(prop, "rna_Mesh_vertex_color_active_index_get",
                                   "rna_Mesh_vertex_color_active_index_set", "rna_Mesh_vertex_color_index_range");
        RNA_def_property_ui_text(prop, "Active Vertex Color Index", "Active vertex color index");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 +      RNA_def_property_update(prop, 0, "rna_Mesh_update_data_edit_active_color");
  }
  
  static void rna_def_uv_layers(BlenderRNA *brna, PropertyRNA *cprop)
        StructRNA *srna;
        PropertyRNA *prop;
  
 -      /* FunctionRNA *func; */
 -      /* PropertyRNA *parm; */
 +      FunctionRNA *func;
 +      PropertyRNA *parm;
  
        RNA_def_property_srna(cprop, "UVLoopLayers");
        srna = RNA_def_struct(brna, "UVLoopLayers", NULL);
        RNA_def_struct_sdna(srna, "Mesh");
        RNA_def_struct_ui_text(srna, "UV Loop Layers", "Collection of uv loop layers");
  
 +      func = RNA_def_function(srna, "new", "rna_Mesh_uv_layers_new");
 +      RNA_def_function_ui_description(func, "Add a UV map layer to Mesh");
 +      RNA_def_string(func, "name", "UVMap", 0, "", "UV map name");
 +      parm = RNA_def_pointer(func, "layer", "MeshUVLoopLayer", "", "The newly created layer");
 +      RNA_def_parameter_flags(parm, 0, PARM_RNAPTR);
 +      RNA_def_function_return(func, parm);
 +
 +      func = RNA_def_function(srna, "remove", "rna_Mesh_uv_layers_remove");
 +      RNA_def_function_ui_description(func, "Remove a vertex color layer");
 +      RNA_def_function_flag(func, FUNC_USE_REPORTS);
 +      parm = RNA_def_pointer(func, "layer", "MeshUVLoopLayer", "", "The layer to remove");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +
        prop = RNA_def_property(srna, "active", PROP_POINTER, PROP_NONE);
        RNA_def_property_struct_type(prop, "MeshUVLoopLayer");
        RNA_def_property_pointer_funcs(prop, "rna_Mesh_uv_layer_active_get",
@@@ -3077,6 -3114,7 +3077,6 @@@ static void rna_def_tessface_uv_texture
        RNA_def_parameter_flags(parm, 0, PARM_RNAPTR);
        RNA_def_function_return(func, parm);
  
 -
        prop = RNA_def_property(srna, "active", PROP_POINTER, PROP_NONE);
        RNA_def_property_struct_type(prop, "MeshTextureFaceLayer");
        RNA_def_property_pointer_funcs(prop, "rna_Mesh_tessface_uv_texture_active_get",
  
        prop = RNA_def_property(srna, "active_index", PROP_INT, PROP_UNSIGNED);
        RNA_def_property_int_funcs(prop, "rna_Mesh_tessface_uv_texture_active_index_get",
 -                                 "rna_Mesh_tessface_uv_texture_active_index_set", "rna_Mesh_uv_texture_index_range");
 -      RNA_def_property_ui_text(prop, "Active UV Map Index", "Active UV Map index");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 -}
 -
 -
 -static void rna_def_uv_textures(BlenderRNA *brna, PropertyRNA *cprop)
 -{
 -      StructRNA *srna;
 -      PropertyRNA *prop;
 -
 -      FunctionRNA *func;
 -      PropertyRNA *parm;
 -
 -      RNA_def_property_srna(cprop, "UVTextures");
 -      srna = RNA_def_struct(brna, "UVTextures", NULL);
 -      RNA_def_struct_sdna(srna, "Mesh");
 -      RNA_def_struct_ui_text(srna, "UV Maps", "Collection of UV maps");
 -
 -      func = RNA_def_function(srna, "new", "rna_Mesh_uv_texture_new");
 -      RNA_def_function_ui_description(func, "Add a UV map layer to Mesh");
 -      RNA_def_string(func, "name", "UVMap", 0, "", "UV map name");
 -      parm = RNA_def_pointer(func, "layer", "MeshTexturePolyLayer", "", "The newly created layer");
 -      RNA_def_parameter_flags(parm, 0, PARM_RNAPTR);
 -      RNA_def_function_return(func, parm);
 -
 -      func = RNA_def_function(srna, "remove", "rna_Mesh_uv_texture_layers_remove");
 -      RNA_def_function_ui_description(func, "Remove a vertex color layer");
 -      RNA_def_function_flag(func, FUNC_USE_REPORTS);
 -      parm = RNA_def_pointer(func, "layer", "MeshTexturePolyLayer", "", "The layer to remove");
 -      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 -
 -      prop = RNA_def_property(srna, "active", PROP_POINTER, PROP_NONE);
 -      RNA_def_property_struct_type(prop, "MeshTexturePolyLayer");
 -      RNA_def_property_pointer_funcs(prop, "rna_Mesh_uv_texture_active_get",
 -                                     "rna_Mesh_uv_texture_active_set", NULL, NULL);
 -      RNA_def_property_flag(prop, PROP_EDITABLE | PROP_NEVER_UNLINK);
 -      RNA_def_property_ui_text(prop, "Active UV Map", "Active UV Map");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 -
 -      prop = RNA_def_property(srna, "active_index", PROP_INT, PROP_UNSIGNED);
 -      RNA_def_property_int_funcs(prop, "rna_Mesh_uv_texture_active_index_get",
 -                                 "rna_Mesh_uv_texture_active_index_set", "rna_Mesh_uv_texture_index_range");
 +                                 "rna_Mesh_tessface_uv_texture_active_index_set", "rna_Mesh_uv_layer_index_range");
        RNA_def_property_ui_text(prop, "Active UV Map Index", "Active UV Map index");
        RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
  }
@@@ -3169,79 -3249,6 +3169,79 @@@ static void rna_def_paint_mask(BlenderR
        RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
  }
  
 +static void rna_def_face_map(BlenderRNA *brna)
 +{
 +      StructRNA *srna;
 +      PropertyRNA *prop;
 +
 +      srna = RNA_def_struct(brna, "MeshFaceMapLayer", NULL);
 +      RNA_def_struct_ui_text(srna, "Mesh Face Map Layer", "Per-face map index");
 +      RNA_def_struct_sdna(srna, "CustomDataLayer");
 +      RNA_def_struct_path_func(srna, "rna_MeshFaceMapLayer_path");
 +
 +      prop = RNA_def_property(srna, "name", PROP_STRING, PROP_NONE);
 +      RNA_def_struct_name_property(srna, prop);
 +      RNA_def_property_string_funcs(prop, NULL, NULL, "rna_MeshPolyLayer_name_set");
 +      RNA_def_property_ui_text(prop, "Name", "Name of face-map layer");
 +      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 +
 +      prop = RNA_def_property(srna, "data", PROP_COLLECTION, PROP_NONE);
 +      RNA_def_property_struct_type(prop, "MeshFaceMap");
 +      RNA_def_property_ui_text(prop, "Data", "");
 +      RNA_def_property_collection_funcs(prop, "rna_MeshFaceMapLayer_data_begin", "rna_iterator_array_next",
 +                                        "rna_iterator_array_end", "rna_iterator_array_get",
 +                                        "rna_MeshFaceMapLayer_data_length", NULL, NULL, NULL);
 +
 +      /* FaceMap struct */
 +      srna = RNA_def_struct(brna, "MeshFaceMap", NULL);
 +      RNA_def_struct_sdna(srna, "MIntProperty");
 +      RNA_def_struct_ui_text(srna, "Int Property", "");
 +      RNA_def_struct_path_func(srna, "rna_MeshFaceMap_path");
 +
 +      prop = RNA_def_property(srna, "value", PROP_INT, PROP_NONE);
 +      RNA_def_property_int_sdna(prop, NULL, "i");
 +      RNA_def_property_ui_text(prop, "Value", "");
 +      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 +}
 +
 +static void rna_def_face_maps(BlenderRNA *brna, PropertyRNA *cprop)
 +{
 +      StructRNA *srna;
 +      PropertyRNA *prop;
 +
 +      RNA_def_property_srna(cprop, "MeshFaceMapLayers");
 +      srna = RNA_def_struct(brna, "MeshFaceMapLayers", NULL);
 +      RNA_def_struct_ui_text(srna, "Mesh Face Map Layer", "Per-face map index");
 +      RNA_def_struct_sdna(srna, "Mesh");
 +      RNA_def_struct_ui_text(srna, "Mesh FaceMaps", "Collection of mesh face-maps");
 +
 +      /* add this since we only ever have one layer anyway, don't bother with active_index */
 +      prop = RNA_def_property(srna, "active", PROP_POINTER, PROP_NONE);
 +      RNA_def_property_struct_type(prop, "MeshFaceMapLayer");
 +      RNA_def_property_pointer_funcs(prop, "rna_Mesh_face_map_active_get",
 +                                     NULL, NULL, NULL);
 +      RNA_def_property_ui_text(prop, "Active FaceMap Layer", "");
 +      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 +
 +      FunctionRNA *func;
 +      PropertyRNA *parm;
 +
 +      func = RNA_def_function(srna, "new", "rna_Mesh_face_map_new");
 +      RNA_def_function_flag(func, FUNC_USE_REPORTS);
 +      RNA_def_function_ui_description(func, "Add a float property layer to Mesh");
 +      RNA_def_string(func, "name", "Face Map", 0, "", "Face map name");
 +      parm = RNA_def_pointer(func, "layer", "MeshFaceMapLayer", "", "The newly created layer");
 +      RNA_def_parameter_flags(parm, 0, PARM_RNAPTR);
 +      RNA_def_function_return(func, parm);
 +
 +      func = RNA_def_function(srna, "remove", "rna_Mesh_face_map_remove");
 +      RNA_def_function_ui_description(func, "Remove a face map layer");
 +      RNA_def_function_flag(func, FUNC_USE_REPORTS);
 +      parm = RNA_def_pointer(func, "layer", "MeshFaceMapLayer", "", "The layer to remove");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +      RNA_def_property_clear_flag(parm, PROP_THICK_WRAP);
 +}
 +
  static void rna_def_mesh(BlenderRNA *brna)
  {
        StructRNA *srna;
                                 "All UV maps for tessellated faces (read-only, for use by renderers)");
        rna_def_tessface_uv_textures(brna, prop);
  
 -      /* UV maps */
 -      prop = RNA_def_property(srna, "uv_textures", PROP_COLLECTION, PROP_NONE);
 -      RNA_def_property_collection_sdna(prop, NULL, "pdata.layers", "pdata.totlayer");
 -      RNA_def_property_collection_funcs(prop, "rna_Mesh_uv_textures_begin", NULL, NULL, NULL,
 -                                        "rna_Mesh_uv_textures_length", NULL, NULL, NULL);
 -      RNA_def_property_struct_type(prop, "MeshTexturePolyLayer");
 -      RNA_def_property_ui_text(prop, "UV Maps", "All UV maps");
 -      rna_def_uv_textures(brna, prop);
 -
 -      prop = RNA_def_property(srna, "uv_texture_clone", PROP_POINTER, PROP_NONE);
 -      RNA_def_property_struct_type(prop, "MeshTexturePolyLayer");
 -      RNA_def_property_pointer_funcs(prop, "rna_Mesh_uv_texture_clone_get",
 -                                     "rna_Mesh_uv_texture_clone_set", NULL, NULL);
 -      RNA_def_property_flag(prop, PROP_EDITABLE);
 -      RNA_def_property_ui_text(prop, "Clone UV Map", "UV map to be used as cloning source");
 -
 -      prop = RNA_def_property(srna, "uv_texture_clone_index", PROP_INT, PROP_UNSIGNED);
 -      RNA_def_property_int_funcs(prop, "rna_Mesh_uv_texture_clone_index_get",
 -                                 "rna_Mesh_uv_texture_clone_index_set", "rna_Mesh_uv_texture_index_range");
 -      RNA_def_property_ui_text(prop, "Clone UV Map Index", "Clone UV map index");
 -
 -      prop = RNA_def_property(srna, "uv_texture_stencil", PROP_POINTER, PROP_NONE);
 -      RNA_def_property_struct_type(prop, "MeshTexturePolyLayer");
 -      RNA_def_property_pointer_funcs(prop, "rna_Mesh_uv_texture_stencil_get",
 -                                     "rna_Mesh_uv_texture_stencil_set", NULL, NULL);
 -      RNA_def_property_flag(prop, PROP_EDITABLE);
 -      RNA_def_property_ui_text(prop, "Mask UV Map", "UV map to mask the painted area");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 -
 -      prop = RNA_def_property(srna, "uv_texture_stencil_index", PROP_INT, PROP_UNSIGNED);
 -      RNA_def_property_int_funcs(prop, "rna_Mesh_uv_texture_stencil_index_get",
 -                                 "rna_Mesh_uv_texture_stencil_index_set", "rna_Mesh_uv_texture_index_range");
 -      RNA_def_property_ui_text(prop, "Mask UV Map Index", "Mask UV map index");
 -      RNA_def_property_update(prop, 0, "rna_Mesh_update_data");
 -
        /* Tessellated face colors - used by renderers */
  
        prop = RNA_def_property(srna, "tessface_vertex_colors", PROP_COLLECTION, PROP_NONE);
        RNA_def_property_ui_text(prop, "String Property Layers", "");
        rna_def_polygon_string_layers(brna, prop);
  
 +      /* face-maps */
 +      prop = RNA_def_property(srna, "face_maps", PROP_COLLECTION, PROP_NONE);
 +      RNA_def_property_collection_sdna(prop, NULL, "pdata.layers", "pdata.totlayer");
 +      RNA_def_property_collection_funcs(prop, "rna_Mesh_face_maps_begin", NULL, NULL, NULL,
 +                                        "rna_Mesh_face_maps_length", NULL, NULL, NULL);
 +      RNA_def_property_struct_type(prop, "MeshFaceMapLayer");
 +      RNA_def_property_ui_text(prop, "FaceMap", "");
 +      rna_def_face_maps(brna, prop);
 +
        /* Skin vertices */
        prop = RNA_def_property(srna, "skin_vertices", PROP_COLLECTION, PROP_NONE);
        RNA_def_property_collection_sdna(prop, NULL, "vdata.layers", "vdata.totlayer");
@@@ -3680,10 -3713,10 +3680,10 @@@ void RNA_def_mesh(BlenderRNA *brna
        rna_def_mpolygon(brna);
        rna_def_mloopuv(brna);
        rna_def_mtface(brna);
 -      rna_def_mtexpoly(brna);
        rna_def_mcol(brna);
        rna_def_mloopcol(brna);
        rna_def_mproperties(brna);
 +      rna_def_face_map(brna);
  }
  
  #endif